色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

java本身自帶的SPI擴展機制是怎么一回事?

jf_ro2CN3Fa ? 來源:碼農參上 ? 2024-01-02 10:32 ? 次閱讀

八股文背多了,相信大家都聽說過一個詞,SPI 擴展

有的面試官就很喜歡問這個問題,SpringBoot 的自動裝配是如何實現的?

基本上,你一說是基于 spring 的 SPI 擴展機制,再把spring.factories文件和EnableAutoConfiguration提一下,那么這個問題就答的八九不離十了。

就像四五年前,我去面試的時候被問到這個問題,SPI 動態擴展機制這幾個詞從嘴里一說出來,就把面試官唬的一愣一愣的。可能他們也沒見過這么能裝逼的,一句話能簡簡單單說明白,非要拽一個聽上去很高大上的詞。

話說回來,被唬住的可不止是面試官,其實還有我自己。至于 SPI 擴展究竟是個啥,是怎么實現的,我當時也根本不明白。

不過現在的面試就是這樣,對線八股文,要想唬住面試官,就得先唬住自己。

那么我們今天暫且不提 spring 的 SPI 擴展,先來看看 java 本身自帶的 SPI 擴展機制是怎么一回事。

1、簡介

SPI 的全稱是Service Provider Interface,翻譯過來就是服務提供者的接口,它所實現的其實是一種服務的發現機制。

這么說起來可能還是有點不好理解,我舉個例子來類比一下。

在 spring 項目中,寫 service 層代碼前,會約定俗成的會添加一個接口層。然后通過 spring 中的依賴注入,可以借助@Autowired等方式注入這個接口的實現類的實例對象,之后對于 service 的調用一般也基于接口操作。

簡單形容就是這樣的:

a61bf46c-a6f5-11ee-8b88-92fbcf53809c.jpg

如圖所示,接口、實現類都是由服務提供方提供,我們可以把 controller 看作服務調用者,調用方只管調用接口就可以了。

雖然也有聲音認為,大部分情況下 service 只有一個實現類,接口層顯得有些多余。但是在《Head First Design Patterns》這本書中,大佬們還是建議過:

Program to an interface, not an implementation.

沒錯,就是常說的要面向接口編程。至于好處,也不外乎是降低耦合度、方便日后擴展、提高了代碼的靈活性和可維護性等等。

在上面這個例子里,這個接口層和其中的方法我們可以稱之為API,而我們要討論的SPI和它相比,有類似也有差異,還是先看圖:

a628f11c-a6f5-11ee-8b88-92fbcf53809c.jpg

簡單來說,就是服務的調用方定義一個接口規范,可以由不同的服務提供者實現。并且,調用方能夠通過某種機制來發現服務提供方,并通過接口調用它的能力。

通過對比,我們可以看出它們雖然都有著接口這一層面,但還是有很大的不同:

API 中的接口是服務提供者給服務調用者的一個功能列表,而 SPI 中更多強調的是,服務調用者對服務實現的一種約束,服務提供者根據這種約束實現的服務,可以被服務調用者發現。

說白了,Java 中的 SPI 實現的就是,你按我的接口規范實現服務,我就能通過某種機制為這個接口尋找到這個服務。

這么說起來可能還有些抽象,下面我們舉一個例子,類比具體描述一下這個過程。

2、定義接口

說起智能家居系統,大家現在都比較熟悉了,只要是相同品牌下的產品,連上 wifi 就能夠通過手機 app 控制了,非常方便。

雖然產品不斷更新換代,型號更新層出不窮,但是同種家電在 app 上操作起來,功能一般都是一樣的。就拿空調來說,我們在 app 上操作起來一般也就三個主要功能:開關選模式調節溫度

假設我現在在客廳、臥室、書房安裝了 3 款不同型號的空調,并把它們都接入到了我 app 中,那么之后的操作都是相同的幾個按鍵,簡單粗暴。

a64497aa-a6f5-11ee-8b88-92fbcf53809c.jpg

思考一下,無論是開關還是調溫,都是通過 app 去調用設備的接口罷了,那么如果不同型號的空調各寫各的接口,后端 app 在開發的時候光對接接口都麻煩的要死。

解決方法也很簡單,我先定義一套接口規范,不管你以后什么型號的空調,都按我的規范來實現接口。以后只要我能發現你的設備,那么都可以按相同的方法來調用接口。

那么下面就先來定義這么一套接口規范,如果你以后想要接入智能家居系統,那么就要遵循這個規范來開發接口。

新建一個項目作為標準,就叫aircondition-standard好了,然后創建一個接口。除了 3 個操作以外,我們再添加一個獲取空調型號的方法。

publicinterfaceIAircondition{
//獲取型號
StringgetType();

//開關
voidturnOnOff();

//調節溫度
voidadjustTemperature(inttemperature);

//模式變更
voidchangeModel(intmodelId);
}

這個接口后面要給服務的實現方來使用,用 maven 把它打成 jar 包:

mvncleaninstall

之后服務提供者在項目中就可以引入這個 jar 包了,有了這套規范,就保證了產品后期不管怎么更新換代,都能接入到系統來。

3、服務實現

制定并發布完規則后,掛式空調作為第一個服務提供者就來了,新建一個項目aircondition-hanging-type,并引入剛才打好的 jar 包:


com.cn.hydra
aircondition-standard
1.0-SNAPSHOT

創建服務類,并實現前面定義的接口:

publicclassHangingTypeAircondition
implementsIAircondition{
publicStringgetType(){
return"HangingType";
}

publicvoidturnOnOff(){
System.out.println("掛式空調開關");
}

publicvoidadjustTemperature(inti){
System.out.println("掛式空調調節溫度");
}

publicvoidchangeModel(inti){
System.out.println("掛式空調更換模式");
}
}

在項目的resources的目錄下,創建META-INF/services目錄,然后以前面定義的接口名com.cn.hydra.IAircondition創建文件,并在文件中寫入實現類的全限定名。

com.cn.hydra.HangingTypeAircondition

整個項目結構非常簡單:

a65e6e8c-a6f5-11ee-8b88-92fbcf53809c.png

這樣,一個服務方的簡單實現就搞定了,用 maven 打成 jar 包,之后就可以提供給調用方使用了。

同理,我們可以再創建一個立式空調的項目aircondition-vertical-type,也只創建一個服務類:

publicclassVerticalTypeAircondition
implementsIAircondition{
publicStringgetType(){
return"VerticalType";
}

publicvoidturnOnOff(){
System.out.println("立式空調開關");
}

publicvoidadjustTemperature(inti){
System.out.println("立式空調調節溫度");
}

publicvoidchangeModel(inti){
System.out.println("立式空調更換模式");
}
}

還是按上面的命名規則,創建一個配置文件:

com.cn.hydra.VerticalTypeAircondition

同樣,打成 jar 包就完事了,至于服務調用者如何去發現和調用這兩個服務,下面詳細再說。

4、服務發現

現在兩個服務提供方都實現了接口,下面關鍵的一步就是服務發現,這一步 java 中的 spi 發現機制已經幫我們實現好了。

創建一個新項目aircondition-app,引入上面打好的兩個 jar 包。



com.cn.hydra
aircondition-hanging-type
1.0-SNAPSHOT



com.cn.hydra
aircondition-vertical-type
1.0-SNAPSHOT


按照上面的說法,雖然每個服務提供者對于接口都有不同的實現,但是作為調用者來說,它并不需要關心具體的實現類,我們要做的是通過接口來調用服務提供者實現的方法。

下面,就是關鍵的服務發現環節,我們寫一個方法,根據型號去調用對應空調的開關方法。

publicclassAirconditionApp{
publicstaticvoidmain(String[]args){
newAirconditionApp().turnOn("VerticalType");
}

publicvoidturnOn(Stringtype){
ServiceLoaderload=ServiceLoader
.load(IAircondition.class);

for(IAirconditioniAircondition:load){
System.out.println("檢測到:"+iAircondition.getClass().getSimpleName());
if(type.equals(iAircondition.getType())){
iAircondition.turnOnOff();
}
}
}
}

測試結果:

a677a3fc-a6f5-11ee-8b88-92fbcf53809c.png

可以看到,測試過程中,通過定義的接口IAircondition發現了兩個實現類,并通過參數,調用了特定實現類的某個方法。整段代碼中沒有出現過具體的服務實現類,操作都是通過接口調用。

5、原理

了解了 spi 的工作流程,我們再來看看它的實現,其實最關鍵的就是上面代碼中出現的ServiceLoader這個類。

上面的示例代碼中,對于ServiceLoader的load()方法的結果,我們用for循環進行了遍歷,這一點我們看一下源碼就能明白,因為ServiceLoader實現了Iterable這一接口,而整個服務發現的核心,就在它的iterator()方法中。

a69198d4-a6f5-11ee-8b88-92fbcf53809c.png

注意這里面有兩個關鍵的東西,找一下在源碼中定義的地方:

a695696e-a6f5-11ee-8b88-92fbcf53809c.png

注釋寫的非常明白,providers就是一個緩存,在迭代器中如果先從這里面進行查找,如果里面有就繼續往下找,沒有了的話就用這個懶加載的lookupIterator查找。

那么就簡單了,接著往下看LazyIterator,看看它里面的hasNext()和next()兩個方法是怎么實現的。

a6aba8aa-a6f5-11ee-8b88-92fbcf53809c.png

這個acc是一個安全管理器,在前面通過System.getSecurityManager()判斷并賦值,debug 看一下這里都是null,所以直接看hasNextService()和nextService()方法就可以了。

在hasNextService()方法中,會取出接口取出實現類的類名放到nextName中:

a6c34690-a6f5-11ee-8b88-92fbcf53809c.png

接下來,在nextService()方法中,則會先加載這個實現類,然后實例化對象,最終放入緩存中去。

a6d616c6-a6f5-11ee-8b88-92fbcf53809c.png

在迭代器的迭代過程中,會完成所有實現類的實例化,其實歸根結底,還是基于 java 反射去實現的。

6、應用

要說 spi 的實際應用,大家最常見的應該就是日志框架slf4j了,它利用 spi 實現了插槽式接入其他具體的日志框架。

說白了,slf4j本身就是個日志門面,并不提供具體的實現,需要綁定其他具體實現才能真正的引入日志功能。

例如我們可使用log4j2作為具體的綁定器,只需要在 pom 中引入slf4j-log4j12,就可以使用具體功能。


org.slf4j
slf4j-api
2.0.3


org.slf4j
slf4j-log4j12
2.0.3

引入項目后,點開它的 jar 包看一下具體結構:

a6ef60d6-a6f5-11ee-8b88-92fbcf53809c.png

有沒有發現一個彩蛋,先說為什么我們 pom 中引入的明明是slf4j-log4j12,實際上引入的是slf4j-reload4j?翻一下官網的文檔:

a71e9b76-a6f5-11ee-8b88-92fbcf53809c.png

大意就是在 2015 年和 2022 年,log4j1.x就已經宣布end of life終止了,原因也不難猜,估計是因為頻繁爆出的漏洞。在那之后,slf4j-log4j在構建階段就會自動重定向到slf4j-reload4j了,并且官方也強烈建議使用slf4j-reload4j作為替代。

再回頭看一下 jar 包的META-INF.services里面,通過 spi 注入了Reload4jServiceProvider這個實現類,它實現了SLF4JServiceProvider這一接口,在它的初始化方法initialize()中,會完成初始化等工作,后續可以繼續獲取到LoggerFactory和Logger等具體日志對象。

7、總結

Java 中的 SPI 提供了一種比較特別的服務發現和調用機制,通過接口靈活的將服務調用與服務提供者分離,用于提供給第三方實現擴展時還是很方便的。但是也有缺點,比方說一旦加載一個接口,就會把所有實現類都加載進來,可能會加載到不需要的冗余服務。不過站在整體角度上,還是給我們提供了一種非常不錯的框架擴展、集成的思路。

a74cd8a6-a6f5-11ee-8b88-92fbcf53809c.png






審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • JAVA
    +關注

    關注

    19

    文章

    2969

    瀏覽量

    104783
  • SPI接口
    +關注

    關注

    0

    文章

    258

    瀏覽量

    34387
  • for循環
    +關注

    關注

    0

    文章

    61

    瀏覽量

    2503

原文標題:美團:SPI 的原理是什么?

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    法拉第圓筒是怎么一回事

    法拉第圓筒是怎么一回事啊???求詳細的解說...
    發表于 07-30 14:40

    請問下數碼管陽極顯示和共陽極是一回事嗎?

    本帖最后由 eehome 于 2013-1-5 09:43 編輯 請問下數碼管陽極顯示和共陽極是一回事嗎?
    發表于 12-03 11:21

    藍牙中的拓撲結構有散射網,MESH網是同一回事嗎?

    在藍牙的學習中,有講到藍牙的拓撲結構,包含微微網和散射網,其中散射網是微微網的拓展網絡設備數量的,請問散射網和MESH網是不是一回事,謝謝
    發表于 03-15 19:38

    Keil軟件仿真STM32時出現錯誤是怎樣一回事

    Keil軟件仿真STM32時出現錯誤是怎樣一回事?怎樣去解決這個問題?
    發表于 11-10 06:23

    SFUD庫看不到初始化spi1的GPIO的代碼是怎么一回事

    SFUD庫 看不到初始化spi1的GPIO的代碼是怎么一回事
    發表于 07-29 10:39

    請問電源去耦和電源濾波是一回事嗎?

    請問電源去耦和電源濾波是一回事嗎?
    發表于 04-21 17:42

    請問芯片中ISP Flash和LDROM是不是一回事

    芯片中 ISP Flash 和 LDROM 是不是一回事? 如果不是一回事,以M453VG6AE為例,它們基地址分別是什么?
    發表于 08-29 08:08

    請問KVA和KW是不是一回事

    KVA 和KW是不是一回事? 比如負載時2kw那么我的變壓器的容量需要大于2/cosφ呢?
    發表于 12-11 07:43

    慢速保險絲是怎樣一回事

    慢速保險絲是怎樣一回事? 慢速保險絲也叫延時保險絲,它的延時特性表現在電路出現非故障脈沖電流時保持完好而能對長時間的過載提供保護。有些電路在
    發表于 11-12 09:11 ?881次閱讀

    機器人即服務是怎么一回事

    機器人即服務是怎么一回事
    發表于 08-06 16:48 ?1584次閱讀

    PCB設計中自動布線和手動布線是怎么一回事?資料下載

    電子發燒友網為你提供PCB設計中自動布線和手動布線是怎么一回事?資料下載的電子資料下載,更有其他相關的電路圖、源代碼、課件教程、中文資料、英文資料、參考設計、用戶指南、解決方案等資料,希望可以幫助到廣大的電子工程師們。
    發表于 04-12 08:51 ?14次下載
    PCB設計中自動布線和手動布線是怎么<b class='flag-5'>一回事</b>?資料下載

    手機里的射頻與天線是一回事

    手機里的射頻與天線當然不是一回事了。
    的頭像 發表于 10-04 12:52 ?1.1w次閱讀

    SMT生產過程中拋料是怎么一回事呢?具體需要怎么解決?

    在SMT工廠,生產過程中經常會遇到拋料的情況,甚至有時候拋料會非常嚴重,影響到生產效率,那么拋料是怎么一回事呢?具體需要怎么解決?
    的頭像 發表于 01-24 10:42 ?2924次閱讀

    電機和馬達是一回事嗎 馬達和電機有什么區別

    電機和馬達是一回事嗎 馬達和電機有什么區別 電機和馬達是一回事嗎? 電機和馬達是同個名詞的不同表達方式。在些地區,特別是中國,人們更傾向于使用“電機”來指代電動機,而在其他地區則更
    的頭像 發表于 02-03 09:19 ?1w次閱讀

    電機驅動芯片和電機控制芯片是一回事

    電機驅動芯片和電機控制芯片在電機系統中各自扮演著重要的角色,但它們并不是一回事
    的頭像 發表于 04-08 11:15 ?1662次閱讀
    主站蜘蛛池模板: jizzjizz中国大学生| 中国老女人xxhd69| 久99视频精品免费观看福利| 国产1769一七六九视频在线| 97色伦久久视频在观看| 优优色影院| 亚洲欧美日韩人成| 亚洲精品第一综合99久久| 午夜理伦片免费| 无码乱人伦一区二区亚洲一| 色欲AV精品人妻一区二区三区| 欧美写真视频一区| 青春禁区动漫免费观看| 欧美日韩中文在线字幕视频| 欧美free嫩交hd| 欧美AAAA片免费播放观看| 欧美日韩高清一区二区三区| 免费又黄又硬又爽大片| 男女爽爽无遮挡午夜视频在线观看| 久久亚洲伊人中字综合精品| 久久日本片精品AAAAA国产| 久久无码AV亚洲精品色午夜麻豆| 久久亚洲AV成人无码国产漫画| 久久免费看少妇高潮A片2012| 久久免费视频1| 凌馨baby| 欧美乱妇日本无乱码特黄大片 | 两个客户一起吃我的奶| 旧里番ovaの催○セイ活指导| 蕾丝边娱乐网| 欧美阿v在线免播播放| 清冷受被CAO的合不拢| 日韩欧美中文字幕在线| 无码国产精品高潮久久9 | 国产电影无码午夜在线播放| 国产人妻精品久久久久久很牛| 好紧小嫩嫩水的10p| 快播欧美大片| 人妻体体内射精一区二区| 四虎影视国产精品亚洲精品hd| 迅雷哥在线观看高清|