Java SPI機(jī)制詳細(xì)介紹
為什么需要SPI?
思考一個(gè)場(chǎng)景,我們封裝了一套服務(wù),別人通過(guò)引入我們寫好的包,就可以使用這些接口API,完成相應(yīng)的操作,這本來(lái)沒(méi)有什么問(wèn)題,但是會(huì)存在使用該服務(wù)的實(shí)體有不相同的業(yè)務(wù)需求,需要進(jìn)一步的擴(kuò)展,但是由于api是寫好的,想要擴(kuò)展并非那么的簡(jiǎn)單,如果存在這樣子的場(chǎng)景,我們?cè)撛趺崔k?
可以使用Java 提供的SPI機(jī)制
什么是SPI?SPI和API的區(qū)別
SPI
SPI的全稱是Service Provider Interface,是Java提供的可用于第三方實(shí)現(xiàn)和擴(kuò)展的機(jī)制,通過(guò)該機(jī)制,我們可以實(shí)現(xiàn)解耦,SPI接口方負(fù)責(zé)定義和提供默認(rèn)實(shí)現(xiàn),SPI調(diào)用方可以按需擴(kuò)展
API的全稱是Application Programming Interface,廣義上來(lái)看就是接口,負(fù)責(zé)程序與程序之間進(jìn)行協(xié)作的通道,就好比上面給的例子,【我們封裝好了一套服務(wù),通過(guò)API的形式提供給他人使用,別人使用API就能得到想要的】
所以他們倆的區(qū)別就很明顯了,API的調(diào)用方只能依賴使用提供方的實(shí)現(xiàn),SPI就如同可定制化的API一樣,調(diào)用方可以自定義實(shí)現(xiàn)替換API提供的默認(rèn)實(shí)現(xiàn)
來(lái)人,上點(diǎn)對(duì)抗
首先,我們新建一個(gè)空的maven項(xiàng)目,里邊有兩個(gè)包
spi-provider從名字就可以得知是SPI的提供方
spi-user SPI的使用方

spi-provider
我們簡(jiǎn)單定義一個(gè)SPI接口,就叫ISpiTest,里邊就一個(gè)saySomething方法,再提供一個(gè)默認(rèn)的實(shí)現(xiàn)
public interface ISpiTest {
void saySomething();
}
public class DefaultSpiImplementation implements ISpiTest{
@Override
public void saySomething() {
System.out.println("[默認(rèn)實(shí)現(xiàn)] -> 今天也是充滿希望的一天");
}
}
然后,模擬走流程,注意步驟4是我們之后要自定義替換的
/**
* 模擬一套流程
* @author Amg
* @date 2021/12/9
*/
public class TestUtils {
public static void workFlow(ISpiTest s) {
System.out.println("1、步驟1.......");
System.out.println("2、步驟2.......");
System.out.println("3、步驟3.......");
System.out.print("4、步驟4:");
s.saySomething();
System.out.println("5、步驟5.......");
}
}
接著,重點(diǎn)來(lái)了,我們需要在resources目錄下面創(chuàng)建/META-INF/services文件夾,然后以SPI接口的全限定類名作為名稱創(chuàng)建一個(gè)文件

往文件里面填寫實(shí)現(xiàn)類的全限定類名,如下
com.amg.spi.DefaultSpiImplementation
到此,spi-provider這個(gè)模塊就完成了,至于之后要怎么使用,到spi-user模塊中進(jìn)一步說(shuō)明
spi-user
首先,我們?cè)趐om文件中,引入spi-provider坐標(biāo)依賴

?然后定義main方法,在main方法中調(diào)用在spi-provider中定義的SPI接口,此時(shí)采用的是默認(rèn)的配置

可以注意到我們使用ServiceLoader這個(gè)類的load方法,傳入SPI接口的字節(jié)碼進(jìn)行構(gòu)造,我們?cè)趕pi-provider中resources中給出了一個(gè)默認(rèn)實(shí)現(xiàn),但是我們是在spi-user中去調(diào)用的,ServiceLoader會(huì)自動(dòng)讀取META-INF下的配置文件,就算是跨jar包也是可以的
然后現(xiàn)在我們?cè)趕pi-user中定義一個(gè)實(shí)現(xiàn)類,以及把他配置到META-INF下(需要注意,這個(gè)配置的全限定類名仍然需要是spi-provider中定義SPI接口的路徑),來(lái)看看效果

spi-user下META-INF里邊內(nèi)容如下
com.amg.spiuser.service.impl.WantHamburger

可以發(fā)現(xiàn),我們并沒(méi)有改變?nèi)魏蔚目蛻舳舜a,只是把配置文件進(jìn)行了簡(jiǎn)單的修改,即可完成自定義實(shí)現(xiàn),這就是使用SPI的魅力
??思考一下,我們之前的流程是怎么做的
首先定義了一個(gè)接口,面向接口編程嘛定義配置文件各個(gè)自定義的實(shí)現(xiàn)類,只需要按照規(guī)則重寫配置文件即可
總結(jié)
通過(guò)這個(gè)流程,我們可以歸納為一句話,SPI是策略模式的一種體現(xiàn),配合面向接口編程的思想以及必要的配置文件,即可完成定義和具體實(shí)現(xiàn)的解耦,而且是可定制化的API
SPI的優(yōu)點(diǎn)有以下
定制化實(shí)現(xiàn)接口解耦
SPI的缺點(diǎn)有以下
通過(guò)觀察ServiceLoader,可以發(fā)現(xiàn)并沒(méi)有額外的加鎖機(jī)制,所以會(huì)存在并發(fā)問(wèn)題獲取對(duì)應(yīng)的實(shí)現(xiàn)類不夠靈活,從上面例子可以看出,需要使用迭代器的方式獲取需要知道接口的所有具體實(shí)現(xiàn)類,所以每次都要加載和實(shí)例化所有的實(shí)現(xiàn)類
實(shí)際中,SPI的使用還是很常見(jiàn)的,例如Dubbo和Spring Boot都為我們提供了一套SPI機(jī)制,只不過(guò)此SPI是在Java提供的SPI機(jī)制基礎(chǔ)上進(jìn)行改造而來(lái),有興趣的同學(xué)也可以去查下資料,增長(zhǎng)增長(zhǎng)
到此這篇關(guān)于Java SPI機(jī)制詳細(xì)介紹的文章就介紹到這了,更多相關(guān)Java SPI機(jī)制內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot集成Milvus實(shí)現(xiàn)數(shù)據(jù)增刪改查功能
milvus支持的語(yǔ)言比較多,支持python, Java, Go,node等開發(fā)語(yǔ)言,本文主要介紹如何使用Java語(yǔ)言,采用springboot框架集成和調(diào)用Milvus數(shù)據(jù)庫(kù),這篇文章主要介紹了SpringBoot集成Milvus,實(shí)現(xiàn)數(shù)據(jù)增刪改查,需要的朋友可以參考下2025-04-04
Java8新特性之重復(fù)注解(repeating annotations)淺析
這篇文章主要介紹了Java8新特性之重復(fù)注解(repeating annotations)淺析,這個(gè)新特性只是修改了程序的可讀性,是比較小的一個(gè)改動(dòng),需要的朋友可以參考下2014-06-06
java Collection 之Set使用說(shuō)明
本篇文章小編為大家介紹,java Collection 之Set使用說(shuō)明。需要的朋友參考下2013-04-04
Spring Security @PreAuthorize注解分析
本教程介紹了如何使用 Spring 方法級(jí)安全和 @PreAuthorize 注解來(lái)保護(hù) RestController 方法,通過(guò)這些步驟,您可以確保只有具有適當(dāng)角色或權(quán)限的用戶才能訪問(wèn)特定的 REST API,感興趣的朋友跟隨小編一起看看吧2024-11-11
Spring-Bean創(chuàng)建對(duì)象的步驟方式詳解
在本篇文章里小編給大家分享的是關(guān)于Spring-Bean創(chuàng)建對(duì)象的步驟方式詳解內(nèi)容,有興趣的朋友們跟著學(xué)習(xí)下。2020-02-02
Mybatis實(shí)現(xiàn)動(dòng)態(tài)排序方式
這篇文章主要介紹了Mybatis實(shí)現(xiàn)動(dòng)態(tài)排序方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10
SpringBoot 2.0 整合sharding-jdbc中間件實(shí)現(xiàn)數(shù)據(jù)分庫(kù)分表
這篇文章主要介紹了SpringBoot 2.0 整合sharding-jdbc中間件,實(shí)現(xiàn)數(shù)據(jù)分庫(kù)分表,本文圖文并茂給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下2019-06-06

