JAVA中的SPI思想介紹
1. SPI介紹
SPI全稱Service Provider Interface,是Java提供的一套用來被第三方實(shí)現(xiàn)或者擴(kuò)展的接口,其意義在于為某個接口尋找服務(wù)的實(shí)現(xiàn),主要應(yīng)用在框架中用來尋找組件,提高擴(kuò)展性。
汽車制造是一個比較繁瑣的過程,通常的手段是先規(guī)定汽車各個零部件的生產(chǎn)規(guī)格,各個零部件廠商按照這種規(guī)則去生產(chǎn)合格的零部件。汽車生產(chǎn)商挑選合適的零部件去組裝以出產(chǎn)汽車。汽車某個零部件損壞也不用廢棄掉整個汽車,只需要更換組件即可。
SPI就是用來怎么去尋找汽車零部件的一種機(jī)制,生產(chǎn)規(guī)格就是接口的定義,零部件生產(chǎn)商生產(chǎn)零部件就是遵循接口提供具體的實(shí)現(xiàn),SPI挑選合適的組件進(jìn)行組裝后完成特定的功能,當(dāng)某個組件存在漏洞或問題時可以在不改動代碼的前提下替換組件以提高擴(kuò)展性。
2. SPI規(guī)則
SPI旨在為某個接口尋找服務(wù)的實(shí)現(xiàn),因此在設(shè)計(jì)初期就要規(guī)定好組件的接口,JAVA的SPI機(jī)制使用步驟如下:
定義組件接口(生產(chǎn)規(guī)格)
實(shí)現(xiàn)組件接口(提供組件)
配置組件文件(查找組件)
反射實(shí)例調(diào)用(組裝工作)
3. SPI案例
組件定義工程中定義組件開發(fā)的規(guī)范,即定義組件需要實(shí)現(xiàn)哪些接口組件A工程和組件B工程提供對組件的實(shí)現(xiàn),即實(shí)現(xiàn)組件定義的接口組件應(yīng)用工程挑選合適的組件進(jìn)行組件的運(yùn)用
3.1 組件的定義
在【commons-api】工程中定義組件的規(guī)范,即定義接口,接口名稱為ComponentService,內(nèi)容如下:
public interface ComponentService { /** * 獲取組件的名稱 * @return 組件名稱 */ String getComponentName(); }
3.2 組件的實(shí)現(xiàn)
在【component-A】工程中按照組件定義的規(guī)范提供組件,即實(shí)現(xiàn)組件定義接口,類名稱為ComponentA,內(nèi)容如下:
public class ComponentA implements ComponentService { /** * 組件名稱 */ private static final String COMPONENT_NAME = "組件A"; @Override public String getComponentName() { return COMPONENT_NAME; } }
按照J(rèn)AVA的SPI規(guī)則在【component-A】工程的resource/META-INF/services資源目錄下新建文件,文件名稱為組件接口的全限定名,內(nèi)容為組件實(shí)現(xiàn)的全限定名
在【component-B】工程中也提供對應(yīng)的組件實(shí)現(xiàn),類名稱為ComponentB,內(nèi)容如下:
public class ComponentB implements ComponentService { /** * 組件名稱 */ private static final String COMPONENT_NAME = "組件B"; @Override public String getComponentName() { return COMPONENT_NAME; } }
同樣在【component-B】工程的resource/META-INF/services資源目錄下配置文件
3.3 組件的選用
基于maven構(gòu)建的java工程使用pom文件來編排項(xiàng)目所需要的依賴組件,現(xiàn)在需要用到組件,并且我覺得A組件比B組件更適合我,如是在【component-application】工程的pom中我編排了組件A,內(nèi)容如下:
<dependencies> <dependency> <groupId>com.xxxx</groupId> <artifactId>component-A</artifactId> <version>1.0-SNAPSHOT</version> </dependency> </dependencies>
在【component-application】工程中新建應(yīng)用程序啟動類來運(yùn)用組件,類名稱為Main,內(nèi)容如下:
public class Main { public static void main(String[] args) { // 加載組件 ServiceLoader<ComponentService> components = ServiceLoader.load(ComponentService.class); for (ComponentService component : components) { // 運(yùn)用組件:打印組件名稱 System.out.println(component.getComponentName()); } } }
啟動【component-application】工程的Main類的main方法,結(jié)果如下:
假如某一天我發(fā)現(xiàn)組件A存在很大漏洞,需要更換組件將組件A替換成組件B,只需要在【component-application】工程的pom中去掉組件A的依賴,導(dǎo)入組件B的依賴即可,pom內(nèi)容如下:
<dependencies> <dependency> <groupId>com.xxxx</groupId> <artifactId>component-B</artifactId> <version>1.0-SNAPSHOT</version> </dependency> </dependencies>
無需修改【component-application】工程的具體使用細(xì)節(jié),就可以達(dá)到替換組件的目的,運(yùn)行如下:
4. SPI原理
JAVA提供的SPI機(jī)制主要依靠的是java.util.ServiceLoader類,從SPI案例中入手,進(jìn)入ServiceLoader.load方法一探究竟。
load方法最終會創(chuàng)建一個LazyIterator 的實(shí)例,看到Iterator大概就知道和迭代器有關(guān),繼續(xù)了解一下LazyIterator 是什么
猜想得不錯,LazyIterator 和迭代器有關(guān),它是ServiceLoader的一個內(nèi)部類,實(shí)現(xiàn)了Iterator接口,那只需要重點(diǎn)關(guān)注Iterator接口定義的方法
public boolean hasNext()
public S next()
Iterator接口定義的hasNext方法用于判斷迭代的是否是否還有下一個元素,LazyIterator 的hasNext方法最終是調(diào)用的hasNextService方法,重點(diǎn)研究這個。
通過類加載器獲取到指定目錄下的資源文件配置的組件實(shí)現(xiàn)的全限定名,存放在configs的一個容器變量中,有了組件實(shí)現(xiàn)的全限定名,后面多半就是反射生成對象返回了,繼續(xù)看一下LazyIterator 的next方法。LazyIterator的next方法主要邏輯是在nextService方法中,仔細(xì)分析一下
和剛才的猜想一致,拿到了組件實(shí)現(xiàn)的全限定名通過Class.forName來生成組件對象,所以程序就通過for循環(huán)得到了對象可以進(jìn)行調(diào)用。
5. SPI要求
java.util.ServiceLoader類是這樣來描述自己的
A simple service-provider loading facility.
一個簡單的服務(wù)提供者加載工具
The only requirement enforced by this facility is thatprovider classes must have a zero-argument constructor so that they can beinstantiated during loading
強(qiáng)制執(zhí)行的唯一要求是提供者類必須有一個無參的構(gòu)造函數(shù),以便它們可以在加載過程中實(shí)例化,從Class.forName生成實(shí)例對象就可以看出使用的是無參構(gòu)造
6. SPI應(yīng)用
? SPI的這種組件替換思想很容易讓人想到我們熟知的JDBC規(guī)范。JDBC是JAVA規(guī)定的應(yīng)用程序連接數(shù)據(jù)庫的標(biāo)準(zhǔn),定義了連接數(shù)據(jù)庫的幾個接口,比如Connection、Statement、ResultSet。各個數(shù)據(jù)庫廠商提供自己的JDBC實(shí)現(xiàn),這就是我們所說的數(shù)據(jù)庫驅(qū)動。開發(fā)人員只需要關(guān)心如何使用JDBC的各個接口,不需要學(xué)習(xí)不同廠商的實(shí)現(xiàn),這就是面向接口編程。
JDBC編程分為四個步驟,SPI在驅(qū)動管理器DriverManager中得到了應(yīng)用
Mysql驅(qū)動和SqlServer驅(qū)動都有SPI的配置
在驅(qū)動管理器的loadInitialDrivers方法中就會通過SPI機(jī)制獲取可用的驅(qū)動,loadInitialDrivers方法會在靜態(tài)代碼塊中被調(diào)用。這里并沒有通過全限定名反射實(shí)例化,真正的驅(qū)動注冊是數(shù)據(jù)庫廠商提供的驅(qū)動類中通過靜態(tài)代碼塊將驅(qū)動注冊到驅(qū)動管理器中的registeredDrivers集合變量中的,以MySQL驅(qū)動為例:
在驅(qū)動管理器的getConnection方法中會遍歷SPI查找到的可用驅(qū)動,通過驅(qū)動獲取鏈接,直至鏈接獲取成功才返回。
總結(jié)
到此這篇關(guān)于JAVA中的SPI思想介紹的文章就介紹到這了,更多相關(guān)JAVA SPI思想內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring中@PropertySource注解使用場景解析
這篇文章主要介紹了Spring中@PropertySource注解使用場景解析,@PropertySource注解就是Spring中提供的一個可以加載配置文件的注解,并且可以將配置文件中的內(nèi)容存放到Spring的環(huán)境變量中,需要的朋友可以參考下2023-11-11SpringMVC如何獲取多種類型數(shù)據(jù)響應(yīng)
這篇文章主要介紹了SpringMVC如何獲取多種類型數(shù)據(jù)響應(yīng),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧2023-11-11Spring中的兩種代理JDK和CGLIB的區(qū)別淺談
本篇文章中主要介紹了Spring中的兩種代理JDK和CGLIB的區(qū)別淺談,詳解的介紹了JDK和CGLIB的原理和方法,有需要的朋友可以了解一下2017-04-04Java數(shù)組擴(kuò)容實(shí)現(xiàn)方法解析
這篇文章主要介紹了Java數(shù)組擴(kuò)容實(shí)現(xiàn)方法解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-11-11Java?精煉解讀數(shù)據(jù)結(jié)構(gòu)的鏈表的概念與實(shí)現(xiàn)
鏈表是一種物理存儲單元上非連續(xù)、非順序的存儲結(jié)構(gòu),數(shù)據(jù)元素的邏輯順序是通過鏈表中的指針連接次序?qū)崿F(xiàn)的,每一個鏈表都包含多個節(jié)點(diǎn),節(jié)點(diǎn)又包含兩個部分,一個是數(shù)據(jù)域,一個是引用域2022-03-03