Java的SPI机制
Dubbo等框架使用到必須掌握。
java.sql.Driver 是 Spi,com.mysql.jdbc.Driver 是 Spi 實(shí)現(xiàn),其它的都是 Api。
package org.hadoop.java;public interface IService {public String sayHello(); public String getScheme();
}
package org.hadoop.java;public class HDFSService implements IService {@Overridepublic String sayHello() {return "Hello HDFS!!";}@Overridepublic String getScheme() {return "hdfs";}
}
package org.hadoop.java;public class LocalService implements IService {@Overridepublic String sayHello() {return "Hello Local!!";}@Overridepublic String getScheme() {return "local";}}
package org.hadoop.java;import java.util.ServiceLoader;public class ServiceLoaderTest {public static void main(String[] args) { //need to define related class full name in /META-INF/services/.... ServiceLoader<IService> serviceLoader = ServiceLoader.load(IService.class); for (IService service : serviceLoader) {System.out.println(service.getScheme()+"="+service.sayHello()); } }
}
需要在項(xiàng)目的\src\main\resources\下增加services目錄,然后再增加一個(gè)文件這個(gè)文件是前面的接口的包名+接口名,例如:org.hadoop.java.IService
整個(gè)項(xiàng)目
輸出:
------------------------------------------------------------------------
Building SPITest 1.0-SNAPSHOT
--------------------------------------------------------------------------- exec-maven-plugin:1.2.1:exec (default-cli) @ SPITest ---
hdfs=Hello HDFS!!
local=Hello Local!!
------------------------------------------------------------------------
BUILD SUCCESS
------------------------------------------------------------------------
Total time: 0.570s
Finished at: Mon Mar 27 20:45:01 CST 2017
Final Memory: 5M/123M
------------------------------------------------------------------------
參考:設(shè)計(jì)原則:小議 SPI 和 API
Java spi機(jī)制淺談
java中的SPI機(jī)制
1 SPI機(jī)制簡(jiǎn)介
SPI的全名為Service Provider Interface.大多數(shù)開(kāi)發(fā)人員可能不熟悉,因?yàn)檫@個(gè)是針對(duì)廠商或者插件的。在java.util.ServiceLoader的文檔里有比較詳細(xì)的介紹。簡(jiǎn)單的總結(jié)下java spi機(jī)制的思想。我們系統(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ī)制。有點(diǎn)類似IOC的思想,就是將裝配的控制權(quán)移到程序之外,在模塊化設(shè)計(jì)中這個(gè)機(jī)制尤其重要。
2 SPI具體約定
java 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í)例化,完成模塊的注入。 基于這樣一個(gè)約定就能很好的找到服務(wù)接口的實(shí)現(xiàn)類,而不需要再代碼里制定。jdk提供服務(wù)實(shí)現(xiàn)查找的一個(gè)工具類:java.util.ServiceLoader
java.util.ServiceLoader使用
今天在看Hadoop源代的時(shí)候發(fā)現(xiàn),在FileSystem中用到了Java.util.ServiceLoader這個(gè)類來(lái)從配置文件中加載子類或者接口的實(shí)現(xiàn)類。以前從來(lái)沒(méi)有使用過(guò)這個(gè)類,進(jìn)去大概看了一下具體的實(shí)現(xiàn)。主要是從META-INF/services這個(gè)目錄下的配置文件加載給定接口或者基類的實(shí)現(xiàn),ServiceLoader會(huì)根據(jù)給定的類的full name來(lái)在META-INF/services下面找對(duì)應(yīng)的文件,在這個(gè)文件中定義了所有這個(gè)類的子類或者接口的實(shí)現(xiàn)類,返回一個(gè)實(shí)例。
總結(jié)
以上是生活随笔為你收集整理的Java的SPI机制的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 股权转让需要多少钱
- 下一篇: JavaAgent 实现字节码注入