欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

關(guān)于springboot中的SPI機(jī)制

 更新時(shí)間:2022年01月28日 11:00:35   作者:小manong  
這篇文章主要介紹了springboot中的SPI機(jī)制,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

一、從java類加載機(jī)制說(shuō)起

java中的類加載器負(fù)載加載來(lái)自文件系統(tǒng)、網(wǎng)絡(luò)或者其他來(lái)源的類文件。jvm的類加載器默認(rèn)使用的是雙親委派模式。

三種默認(rèn)的類加載器Bootstrap ClassLoader、Extension ClassLoader和System ClassLoader(Application ClassLoader)每一個(gè)中類加載器都確定了從哪一些位置加載文件。于此同時(shí)我們也可以通過(guò)繼承java.lang.classloader實(shí)現(xiàn)自己的類加載器。

  • Bootstrap ClassLoader:負(fù)責(zé)加載JDK自帶的rt.jar包中的類文件,是所有類加載的父類
  • Extension ClassLoader:負(fù)責(zé)加載java的擴(kuò)展類庫(kù)從jre/lib/ect目錄或者java.ext.dirs系統(tǒng)屬性指定的目錄下加載類,是System ClassLoader的父類加載器
  • System ClassLoader:負(fù)責(zé)從classpath環(huán)境變量中加載類文件

1、雙親委派模型

原理:

當(dāng)一個(gè)類加載器收到類加載任務(wù)時(shí),會(huì)先交給自己的父加載器去完成,因此最終加載任務(wù)都會(huì)傳遞到最頂層的BootstrapClassLoader,只有當(dāng)父加載器無(wú)法完成加載任務(wù)時(shí),才會(huì)嘗試自己來(lái)加載。

具體:

根據(jù)雙親委派模式,在加載類文件的時(shí)候,子類加載器首先將加載請(qǐng)求委托給它的父加載器,父加載器會(huì)檢測(cè)自己是否已經(jīng)加載過(guò)類,如果已經(jīng)加載則加載過(guò)程結(jié)束,如果沒(méi)有加載的話則請(qǐng)求繼續(xù)向上傳遞直Bootstrap ClassLoader。如果請(qǐng)求向上委托過(guò)程中,如果始終沒(méi)有檢測(cè)到該類已經(jīng)加載,則Bootstrap ClassLoader開(kāi)始嘗試從其對(duì)應(yīng)路勁中加載該類文件,如果失敗則由子類加載器繼續(xù)嘗試加載,直至發(fā)起加載請(qǐng)求的子加載器為止。

采用雙親委派模式可以保證類型加載的安全性,不管是哪個(gè)加載器加載這個(gè)類,最終都是委托給頂層的BootstrapClassLoader來(lái)加載的,只有父類無(wú)法加載自己猜嘗試加載,這樣就可以保證任何的類加載器最終得到的都是同樣一個(gè)Object對(duì)象。

protected Class<?> loadClass(String name, boolean resolve) {
? ? synchronized (getClassLoadingLock(name)) {
? ? // 首先,檢查該類是否已經(jīng)被加載,如果從JVM緩存中找到該類,則直接返回
? ? Class<?> c = findLoadedClass(name);
? ? if (c == null) {
? ? ? ? try {
? ? ? ? ? ? // 遵循雙親委派的模型,首先會(huì)通過(guò)遞歸從父加載器開(kāi)始找,
? ? ? ? ? ? // 直到父類加載器是BootstrapClassLoader為止
? ? ? ? ? ? if (parent != null) {
? ? ? ? ? ? ? ? c = parent.loadClass(name, false);
? ? ? ? ? ? } else {
? ? ? ? ? ? ? ? c = findBootstrapClassOrNull(name);
? ? ? ? ? ? }
? ? ? ? } catch (ClassNotFoundException e) {}
? ? ? ? if (c == null) {
? ? ? ? ? ? // 如果還找不到,嘗試通過(guò)findClass方法去尋找
? ? ? ? ? ? // findClass是留給開(kāi)發(fā)者自己實(shí)現(xiàn)的,也就是說(shuō)
? ? ? ? ? ? // 自定義類加載器時(shí),重寫此方法即可
? ? ? ? ? ?c = findClass(name);
? ? ? ? }
? ? }
? ? if (resolve) {
? ? ? ? resolveClass(c);
? ? }
? ? return c;
? ? }
}

2、雙親委派模型缺陷

在雙親委派模型中,子類加載器可以使用父類加載器已經(jīng)加載的類,而父類加載器無(wú)法使用子類加載器已經(jīng)加載的。這就導(dǎo)致了雙親委派模型并不能解決所有的類加載器問(wèn)題。

案例:

Java 提供了很多服務(wù)提供者接口(Service Provider Interface,SPI),允許第三方為這些接口提供實(shí)現(xiàn)。常見(jiàn)的 SPI 有 JDBC、JNDI、JAXP 等,這些SPI的接口由核心類庫(kù)提供,卻由第三方實(shí)現(xiàn),這樣就存在一個(gè)問(wèn)題:SPI 的接口是 Java 核心庫(kù)的一部分,是由BootstrapClassLoader加載的;SPI實(shí)現(xiàn)的Java類一般是由AppClassLoader來(lái)加載的。BootstrapClassLoader是無(wú)法找到 SPI 的實(shí)現(xiàn)類的,因?yàn)樗患虞dJava的核心庫(kù)。它也不能代理給AppClassLoader,因?yàn)樗亲铐攲拥念惣虞d器。也就是說(shuō),雙親委派模型并不能解決這個(gè)問(wèn)題

3、使用線程上下文類加載器(ContextClassLoader)加載

如果不做任何的設(shè)置,Java應(yīng)用的線程的上下文類加載器默認(rèn)就是AppClassLoader。在核心類庫(kù)使用SPI接口時(shí),傳遞的類加載器使用線程上下文類加載器,就可以成功的加載到SPI實(shí)現(xiàn)的類。線程上下文類加載器在很多SPI的實(shí)現(xiàn)中都會(huì)用到。

通常我們可以通過(guò)Thread.currentThread().getClassLoader()和Thread.currentThread().getContextClassLoader()獲取線程上下文類加載器。

4、使用類加載器加載資源文件,比如jar包

類加載器除了加載class外,還有一個(gè)非常重要功能,就是加載資源,它可以從jar包中讀取任何資源文件,比如,ClassLoader.getResources(String name)方法就是用于讀取jar包中的資源文件

//獲取資源的方法
public Enumeration<URL> getResources(String name) throws IOException {
? ? Enumeration<URL>[] tmp = (Enumeration<URL>[]) new Enumeration<?>[2];
? ? if (parent != null) {
? ? ? ? tmp[0] = parent.getResources(name);
? ? } else {
? ? ? ? tmp[0] = getBootstrapResources(name);
? ? }
? ? tmp[1] = findResources(name);
? ? return new CompoundEnumeration<>(tmp);
}

它的邏輯其實(shí)跟類加載的邏輯是一樣的,首先判斷父類加載器是否為空,不為空則委托父類加載器執(zhí)行資源查找任務(wù),直到BootstrapClassLoader,最后才輪到自己查找。而不同的類加載器負(fù)責(zé)掃描不同路徑下的jar包,就如同加載class一樣,最后會(huì)掃描所有的jar包,找到符合條件的資源文件。

// 使用線程上下文類加載器加載資源
public static void main(String[] args) throws Exception{
? ? // Array.class的完整路徑
? ? String name = "java/sql/Array.class";
? ? Enumeration<URL> urls = Thread.currentThread().getContextClassLoader().getResources(name);
? ? while (urls.hasMoreElements()) {
? ? ? ? URL url = urls.nextElement();
? ? ? ? System.out.println(url.toString());
? ? }
}

二、spring中SPI機(jī)制實(shí)現(xiàn)

1、SPI機(jī)制

(1)SPI思想

SPI的全名為Service Provider Interface.這個(gè)是針對(duì)廠商或者插件的。

SPI的思想:系統(tǒng)里抽象的各個(gè)模塊,往往有很多不同的實(shí)現(xiàn)方案,比如日志模塊的方案,xml解析模塊、jdbc模塊的方案等。面向的對(duì)象的設(shè)計(jì)里,我們一般推薦模塊之間基于接口編程,模塊之間不對(duì)實(shí)現(xiàn)類進(jìn)行硬編碼。一旦代碼里涉及具體的實(shí)現(xiàn)類,就違反了可拔插的原則,如果需要替換一種實(shí)現(xiàn),就需要修改代碼。為了實(shí)現(xiàn)在模塊裝配的時(shí)候能不在程序里動(dòng)態(tài)指明,這就需要一種服務(wù)發(fā)現(xiàn)機(jī)制。java spi就是提供這樣的一個(gè)機(jī)制:為某個(gè)接口尋找服務(wù)實(shí)現(xiàn)的機(jī)制

(2)SPI約定

當(dāng)服務(wù)的提供者,提供了服務(wù)接口的一種實(shí)現(xiàn)之后,在jar包的META-INF/services/目錄里同時(shí)創(chuàng)建一個(gè)以服務(wù)接口命名的文件。該文件里就是實(shí)現(xiàn)該服務(wù)接口的具體實(shí)現(xiàn)類。而當(dāng)外部程序裝配這個(gè)模塊的時(shí)候,就能通過(guò)該jar包META-INF/services/里的配置文件找到具體的實(shí)現(xiàn)類名,并裝載實(shí)例化,完成模塊的注入。通過(guò)這個(gè)約定,就不需要把服務(wù)放在代碼中了,通過(guò)模塊被裝配的時(shí)候就可以發(fā)現(xiàn)服務(wù)類了。

2、SPI使用案例

common-logging apache最早提供的日志的門面接口。只有接口,沒(méi)有實(shí)現(xiàn)。具體方案由各提供商實(shí)現(xiàn), 發(fā)現(xiàn)日志提供商是通過(guò)掃描 META-INF/services/org.apache.commons.logging.LogFactory配置文件,通過(guò)讀取該文件的內(nèi)容找到日志提工商實(shí)現(xiàn)類。只要我們的日志實(shí)現(xiàn)里包含了這個(gè)文件,并在文件里制定 LogFactory工廠接口的實(shí)現(xiàn)類即可。

3、springboot中的類SPI擴(kuò)展機(jī)制

在springboot的自動(dòng)裝配過(guò)程中,最終會(huì)加載META-INF/spring.factories文件,而加載的過(guò)程是由SpringFactoriesLoader加載的。從CLASSPATH下的每個(gè)Jar包中搜尋所有META-INF/spring.factories配置文件,然后將解析properties文件,找到指定名稱的配置后返回。

需要注意的是,其實(shí)這里不僅僅是會(huì)去ClassPath路徑下查找,會(huì)掃描所有路徑下的Jar包,只不過(guò)這個(gè)文件只會(huì)在Classpath下的jar包中。

public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";
// spring.factories文件的格式為:key=value1,value2,value3
// 從所有的jar包中找到META-INF/spring.factories文件
// 然后從文件中解析出key=factoryClass類名稱的所有value值
public static List<String> loadFactoryNames(Class<?> factoryClass, ClassLoader classLoader) {
? ? String factoryClassName = factoryClass.getName();
? ? // 取得資源文件的URL
? ? Enumeration<URL> urls = (classLoader != null ? classLoader.getResources(FACTORIES_RESOURCE_LOCATION) : ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION));
? ? List<String> result = new ArrayList<String>();
? ? // 遍歷所有的URL
? ? while (urls.hasMoreElements()) {
? ? ? ? URL url = urls.nextElement();
? ? ? ? // 根據(jù)資源文件URL解析properties文件,得到對(duì)應(yīng)的一組@Configuration類
? ? ? ? Properties properties = PropertiesLoaderUtils.loadProperties(new UrlResource(url));
? ? ? ? String factoryClassNames = properties.getProperty(factoryClassName);
? ? ? ? // 組裝數(shù)據(jù),并返回
? ? ? ? result.addAll(Arrays.asList(StringUtils.commaDelimitedListToStringArray(factoryClassNames)));
? ? }
? ? return result;
}

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java字符串逆序方法詳情

    Java字符串逆序方法詳情

    這篇文章主要介紹了Java字符逆序,字符逆序主要原理就是將一個(gè)字符串str的內(nèi)容顛倒過(guò)來(lái),并輸出,下文操作分享需要的小伙伴可以參考一下
    2022-03-03
  • Java使用正則表達(dá)式實(shí)現(xiàn)找出數(shù)字功能示例

    Java使用正則表達(dá)式實(shí)現(xiàn)找出數(shù)字功能示例

    這篇文章主要介紹了Java使用正則表達(dá)式實(shí)現(xiàn)找出數(shù)字功能,結(jié)合實(shí)例形式分析了Java針對(duì)數(shù)字的匹配查找及非數(shù)字替換操作相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下
    2017-03-03
  • JVM中四種GC算法案例詳解

    JVM中四種GC算法案例詳解

    這篇文章主要介紹了JVM中四種GC算法案例詳解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-09-09
  • Java中的clone方法詳解_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    Java中的clone方法詳解_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    clone顧名思義就是復(fù)制, 在Java語(yǔ)言中, clone方法被對(duì)象調(diào)用,所以會(huì)復(fù)制對(duì)象。下面通過(guò)本文給大家介紹java中的clone方法,感興趣的朋友一起看看吧
    2017-06-06
  • 對(duì)Netty組件的基本介紹

    對(duì)Netty組件的基本介紹

    這篇文章主要介紹了對(duì)Netty組件的基本介紹,Netty是基于Java NIO client-server的網(wǎng)絡(luò)應(yīng)用框架,使用Netty可以快速開(kāi)發(fā)網(wǎng)絡(luò)應(yīng)用,本文涵蓋了netty開(kāi)發(fā)中主要的組件的介紹,需要的朋友可以參考下
    2021-06-06
  • 詳解SpringBoot如何自定義一個(gè)Starter

    詳解SpringBoot如何自定義一個(gè)Starter

    小伙伴們?cè)?jīng)可能都經(jīng)歷過(guò)整天寫著CURD的業(yè)務(wù),都沒(méi)寫過(guò)一些組件相關(guān)的東西,這篇文章記錄一下SpringBoot如何自定義一個(gè)Starter。原理和理論就不用多說(shuō)了,可以在網(wǎng)上找到很多關(guān)于該方面的資料,這里主要分享如何自定義
    2022-11-11
  • Java類的訪問(wèn)權(quán)限關(guān)鍵字用法說(shuō)明

    Java類的訪問(wèn)權(quán)限關(guān)鍵字用法說(shuō)明

    這篇文章主要介紹了Java類的訪問(wèn)權(quán)限關(guān)鍵字用法說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-09-09
  • Springboot實(shí)現(xiàn)動(dòng)態(tài)定時(shí)任務(wù)流程詳解

    Springboot實(shí)現(xiàn)動(dòng)態(tài)定時(shí)任務(wù)流程詳解

    通過(guò)重寫SchedulingConfigurer方法實(shí)現(xiàn)對(duì)定時(shí)任務(wù)的操作,單次執(zhí)行、停止、啟動(dòng)三個(gè)主要的基本功能,動(dòng)態(tài)的從數(shù)據(jù)庫(kù)中獲取配置的定時(shí)任務(wù)cron信息,通過(guò)反射的方式靈活定位到具體的類與方法中
    2022-09-09
  • java雙向循環(huán)鏈表的實(shí)現(xiàn)代碼

    java雙向循環(huán)鏈表的實(shí)現(xiàn)代碼

    這篇文章介紹了java雙向循環(huán)鏈表的實(shí)現(xiàn)代碼,有需要的朋友可以參考一下
    2013-09-09
  • 基于Column注解的columnDefinition用法

    基于Column注解的columnDefinition用法

    這篇文章主要介紹了Column注解的columnDefinition用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10

最新評(píng)論