您好,登錄后才能下訂單哦!
這篇文章將為大家詳細講解有關Java進階之SPI機制的示例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
SPI的英文全稱為Service Provider Interface,字面意思為服務提供者接口,它是jdk提供給“服務提供廠商”或者“插件開發者”使用的接口。
在面向對象的設計中,模塊之間我們一般會采取面向接口編程的方式,而在實際編程過程過程中,API的實現是封裝在jar中,當我們想要換一種實現方法時,還要生成新的jar替換以前的實現類。而通過jdk的SPI機制就可以實現,首先不需要修改原來作為接口的jar的情況下,將原來實現的那個jar替換為另外一種實現的jar即可。
總結一下SPI的思想:在系統的各個模塊中,往往有不同的實現方案,例如日志模塊的方案、xml解析的方案等,為了在裝載模塊的時候不具體指明實現類,我們需要一種服務發現機制,java spi就提供這樣一種機制。有點類似于IoC的思想,將服務裝配的控制權移到程序之外,在模塊化設計時尤其重要。
順便提一下,Java SPI機制在很多大型中間件嗎,例如Dubbo中均有采用,屬于高級Java開發的進階必備知識點,務必要求掌握。
定義服務的通用接口,針對通用的服務接口,提供具體的實現類。
1.在jar包(服務提供者)的META-INF/services/目錄中,新建一個文件,文件名為SPI接口的"全限定名"。 文件內容為該接口的具體實現類的"全限定名"
2.將spi所在jar放在主程序的classpath中
3.服務調用方使用java.util.ServiceLoader
去動態加載具體的實現類到JVM中
java.sql.Driver的spi實現,有mysql驅動、oracle驅動等。以mysql為例,實現類是com.mysql.jdbc.Driver,在mysql-connector-java-5.1.6.jar中,我們可以看到有一個META-INF/services目錄,目錄下有一個文件名為java.sql.Driver的文件,其中的內容是com.mysql.jdbc.Driver。
在我們使用JDBC獲取連接時,我們通常會調用DriverManager.getConnection()
方法獲取連接對象,而在Driver類初始化時會使用ServiceLoader
動態獲取classpath下“注冊”的驅動實現:
static { loadInitialDrivers(); println("JDBC DriverManager initialized"); } private static void loadInitialDrivers() { ..... AccessController.doPrivileged(new PrivilegedAction<Void>() { public Void run() { //使用ServiceLoader動態加載具體的驅動實現類 ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class); Iterator<Driver> driversIterator = loadedDrivers.iterator(); try{ while(driversIterator.hasNext()) { driversIterator.next(); } } catch(Throwable t) { // Do nothing } return null; } }); ..... }
slf4j是一個典型的門面接口,早起我們使用log4j作為日記記錄框架,我們需要同時引入slf4j和log4j的依賴。后面比較流行logback,我們也想要把項目切換到logback上來,此時利用SPI的機制,我們只需要把log4j的jar包替換為logback的jar包就可以了。
在log4j-to-slf4j.jar
中我們可以看到前面提到的服務提供方的SPI接口聲明:
這樣我們只需要將log4j-to-slf4j.jar
引入classpath,slf4j就能夠獲取到org.apache.logging.slf4j.SLF4JProvider
作為實現類。
模塊依賴關系:
在spi-operate-service
中定義INumOperate
接口:
package cn.bigcoder.spi.operate.service; /** * @author: Jindong.Tian * @date: 2020-11-29 * @description: **/ public interface INumOperate { int exec(int num1, int num2); }
在spi-operate-add
中定義加法實現:
package cn.bigcoder.spi.operate.add; import cn.bigcoder.spi.operate.service.INumOperate; /** * @author: Jindong.Tian * @date: 2020-11-29 * @description: **/ public class NumAddOperateImpl implements INumOperate { public int exec(int num1, int num2) { return num1 + num2; } }
在resource/METAINF/resoures
目錄下創建cn.bigcoder.spi.operate.service.INumOperate
文件:
cn.bigcoder.spi.operate.add.NumAddOperateImpl
在spi-operate-multiplication
模塊中定義乘法的實現:
package cn.bigcoder.spi.operate.multiplication; import cn.bigcoder.spi.operate.service.INumOperate; /** * @author: Jindong.Tian * @date: 2020-11-29 * @description: **/ public class NumMultiOperateImpl implements INumOperate { public int exec(int num1, int num2) { return num1 * num2; } }
同樣的在resource/METAINF/resoures
目錄下創建cn.bigcoder.spi.operate.service.INumOperate
文件:
cn.bigcoder.spi.operate.multiplication.NumMultiOperateImpl
在spi-operate-consumer
模塊中,我們編寫測試類獲取classpath中INumOperate
實現類:
package cn.bigcoder.spi.operate; import cn.bigcoder.spi.operate.service.INumOperate; import org.junit.Test; import java.util.Iterator; import java.util.ServiceLoader; /** * @author: Jindong.Tian * @date: 2020-11-29 * @description: **/ public class INumOperateTest { @Test public void test() { // SPI機制,尋找所有的實現類,順序執行 ServiceLoader<INumOperate> loader = ServiceLoader.load(INumOperate.class); Iterator<INumOperate> iterator = loader.iterator(); if (iterator.hasNext()) { INumOperate numOperate = iterator.next(); System.out.println(numOperate.exec(1, 2)); } else { throw new RuntimeException("classpath中未找到cn.bigcoder.spi.operate.INumOperate實現類"); } } }
此時如果我們在spi-operate-consumer
中引入spi-operate-add
,則測試方法執行求和操作;如果引入spi-operate-multiplication
,則測試方法執行乘法操作。
Java主要應用于:1. web開發;2. Android開發;3. 客戶端開發;4. 網頁開發;5. 企業級應用開發;6. Java大數據開發;7.游戲開發等。
關于“Java進階之SPI機制的示例分析”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。