Java中的SPI機(jī)制案例分享
1 簡(jiǎn)單介紹
當(dāng)我們封裝了一套接口,其它項(xiàng)目想要調(diào)用我們的接口只需要引入我們寫好的包,但是其它項(xiàng)目如果想要對(duì)我們的接口進(jìn)行擴(kuò)展,由于接口是被封裝在依賴包中的,想要擴(kuò)展并不容易,這時(shí)就需要依賴于Java為我們提供的SPI機(jī)制。
SPI的全稱是Service Provider Interface,服務(wù)提供者接口,而與之最接近的概念就是API,全稱Application Programming Interface,應(yīng)用程序編程接口。那么這兩者主要的區(qū)別是什么呢?
API的調(diào)用方只能依賴使用提供方已有的實(shí)現(xiàn),而SPI就是可定制化的API,調(diào)用方可以自定義實(shí)現(xiàn)替換API提供的默認(rèn)實(shí)現(xiàn)。
SPI機(jī)制是非常重要的,尤其是對(duì)于框架來說,它可以用來啟用框架擴(kuò)展和替換組件,我們?cè)谧x框架源碼時(shí)也會(huì)看到大量的SPI的應(yīng)用。
SPI的作用就是為這些被擴(kuò)展的API尋找服務(wù)實(shí)現(xiàn)。
2 SPI 案例
創(chuàng)建一個(gè)工程,一個(gè)做SPI的服務(wù)提供者,一個(gè)做SPI服務(wù)引入擴(kuò)展的測(cè)試,此案例構(gòu)建最簡(jiǎn)單的Maven子父工程即可,在spi-test工程引入spi-provider的依賴。

spi-test添加依賴
<dependencies>
<dependency>
<groupId>org.example</groupId>
<artifactId>spi-provider</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
</dependencies>在spi-provider中,提供接口和一個(gè)默認(rèn)的實(shí)現(xiàn)類

在資源文件中加上如下圖文件,并在改文件中指定接口的默認(rèn)實(shí)現(xiàn)類

spi-test中構(gòu)建可執(zhí)行的測(cè)試方法,直接執(zhí)行,會(huì)拿到默認(rèn)的實(shí)現(xiàn)
我們可以在spi-test中擴(kuò)展,這個(gè)接口,如下圖所示:

這時(shí)我們的擴(kuò)展并不能生效,我們?nèi)孕枰谫Y源文件下,為接口指定我們擴(kuò)展的實(shí)現(xiàn)類,這時(shí)在運(yùn)行上述測(cè)試方法即可得到我們擴(kuò)展后的結(jié)果,這就是SPI機(jī)制。

3 SPI 的原理剖析
ServiceLoader這個(gè)類包含了SPI的核心原理,從開頭指定的路徑前綴,我們就能猜到為什么我們必須將文件放入這個(gè)路徑下才會(huì)生效。
通過load方法,創(chuàng)建一個(gè)ServiceLoader的對(duì)象,進(jìn)入其構(gòu)造方法,進(jìn)行reload, 會(huì)創(chuàng)建一個(gè)新的LazyIterator迭代器,LazyIterator是一個(gè)內(nèi)部類,它負(fù)責(zé)掃描META-INF/services/下的配置文件,并parse所有接口的名字,然后通過全限定類名通過反射進(jìn)行實(shí)現(xiàn)類的加載。
public final class ServiceLoader<S>
implements Iterable<S>
{
// 掃描路徑前綴
private static final String PREFIX = "META-INF/services/";
// 被加載的類或者接口
private final Class<S> service;
// 用于定位、加載和實(shí)例化需要加載類的類加載器
private final ClassLoader loader;
// 上下文對(duì)象
private final AccessControlContext acc;
// 按照實(shí)例化的順序緩存已經(jīng)實(shí)例化的類
private LinkedHashMap<String,S> providers = new LinkedHashMap<>();
// 懶查找迭代器
private LazyIterator lookupIterator;
// 重新加載
public void reload() {
// 清理緩存
providers.clear();
// 新的懶查找迭代器
lookupIterator = new LazyIterator(service, loader);
}
// 構(gòu)造方法
private ServiceLoader(Class<S> svc, ClassLoader cl) {
service = Objects.requireNonNull(svc, "Service interface cannot be null");
loader = (cl == null) ? ClassLoader.getSystemClassLoader() : cl;
acc = (System.getSecurityManager() != null) ? AccessController.getContext() : null;
reload();
}
private static void fail(Class<?> service, String msg, Throwable cause)
throws ServiceConfigurationError
{
throw new ServiceConfigurationError(service.getName() + ": " + msg,
cause);
}
private static void fail(Class<?> service, String msg)
throws ServiceConfigurationError
{
throw new ServiceConfigurationError(service.getName() + ": " + msg);
}
private static void fail(Class<?> service, URL u, int line, String msg)
throws ServiceConfigurationError
{
fail(service, u + ":" + line + ": " + msg);
}
private int parseLine(Class<?> service, URL u, BufferedReader r, int lc,
List<String> names)
throws IOException, ServiceConfigurationError
{
String ln = r.readLine();
if (ln == null) {
return -1;
}
int ci = ln.indexOf('#');
if (ci >= 0) ln = ln.substring(0, ci);
ln = ln.trim();
int n = ln.length();
if (n != 0) {
if ((ln.indexOf(' ') >= 0) || (ln.indexOf('\t') >= 0))
fail(service, u, lc, "Illegal configuration-file syntax");
int cp = ln.codePointAt(0);
if (!Character.isJavaIdentifierStart(cp))
fail(service, u, lc, "Illegal provider-class name: " + ln);
for (int i = Character.charCount(cp); i < n; i += Character.charCount(cp)) {
cp = ln.codePointAt(i);
if (!Character.isJavaIdentifierPart(cp) && (cp != '.'))
fail(service, u, lc, "Illegal provider-class name: " + ln);
}
if (!providers.containsKey(ln) && !names.contains(ln))
names.add(ln);
}
return lc + 1;
}
private Iterator<String> parse(Class<?> service, URL u)
throws ServiceConfigurationError
{
InputStream in = null;
BufferedReader r = null;
ArrayList<String> names = new ArrayList<>();
try {
in = u.openStream();
r = new BufferedReader(new InputStreamReader(in, "utf-8"));
int lc = 1;
while ((lc = parseLine(service, u, r, lc, names)) >= 0);
} catch (IOException x) {
fail(service, "Error reading configuration file", x);
} finally {
try {
if (r != null) r.close();
if (in != null) in.close();
} catch (IOException y) {
fail(service, "Error closing configuration file", y);
}
}
return names.iterator();
}
private class LazyIterator
implements Iterator<S>
{
Class<S> service;
ClassLoader loader;
Enumeration<URL> configs = null;
Iterator<String> pending = null;
String nextName = null;
private LazyIterator(Class<S> service, ClassLoader loader) {
this.service = service;
this.loader = loader;
}
private boolean hasNextService() {
if (nextName != null) {
return true;
}
if (configs == null) {
try {
String fullName = PREFIX + service.getName();
if (loader == null)
configs = ClassLoader.getSystemResources(fullName);
else
configs = loader.getResources(fullName);
} catch (IOException x) {
fail(service, "Error locating configuration files", x);
}
}
while ((pending == null) || !pending.hasNext()) {
if (!configs.hasMoreElements()) {
return false;
}
pending = parse(service, configs.nextElement());
}
nextName = pending.next();
return true;
}
private S nextService() {
if (!hasNextService())
throw new NoSuchElementException();
String cn = nextName;
nextName = null;
Class<?> c = null;
try {
c = Class.forName(cn, false, loader);
} catch (ClassNotFoundException x) {
fail(service,
"Provider " + cn + " not found");
}
if (!service.isAssignableFrom(c)) {
fail(service,
"Provider " + cn + " not a subtype");
}
try {
S p = service.cast(c.newInstance());
providers.put(cn, p);
return p;
} catch (Throwable x) {
fail(service,
"Provider " + cn + " could not be instantiated",
x);
}
throw new Error(); // This cannot happen
}
public boolean hasNext() {
if (acc == null) {
return hasNextService();
} else {
PrivilegedAction<Boolean> action = new PrivilegedAction<Boolean>() {
public Boolean run() { return hasNextService(); }
};
return AccessController.doPrivileged(action, acc);
}
}
public S next() {
if (acc == null) {
return nextService();
} else {
PrivilegedAction<S> action = new PrivilegedAction<S>() {
public S run() { return nextService(); }
};
return AccessController.doPrivileged(action, acc);
}
}
public void remove() {
throw new UnsupportedOperationException();
}
}
public Iterator<S> iterator() {
return new Iterator<S>() {
Iterator<Map.Entry<String,S>> knownProviders
= providers.entrySet().iterator();
public boolean hasNext() {
if (knownProviders.hasNext())
return true;
return lookupIterator.hasNext();
}
public S next() {
if (knownProviders.hasNext())
return knownProviders.next().getValue();
return lookupIterator.next();
}
public void remove() {
throw new UnsupportedOperationException();
}
};
}
public static <S> ServiceLoader<S> load(Class<S> service) {
ClassLoader cl = Thread.currentThread().getContextClassLoader();
return ServiceLoader.load(service, cl);
}
public static <S> ServiceLoader<S> loadInstalled(Class<S> service) {
ClassLoader cl = ClassLoader.getSystemClassLoader();
ClassLoader prev = null;
while (cl != null) {
prev = cl;
cl = cl.getParent();
}
return ServiceLoader.load(service, prev);
}
public String toString() {
return "java.util.ServiceLoader[" + service.getName() + "]";
}
}從上面的源碼中,我們不難發(fā)現(xiàn)ServiceLoader并沒有額外的加鎖機(jī)制,所以會(huì)存在并發(fā)問題,再就是獲取對(duì)應(yīng)的實(shí)現(xiàn)類不夠靈活,需要使用迭代器的方式獲取已知接口的所有具體實(shí)現(xiàn)類,所以每次都要加載和實(shí)例化所有的實(shí)現(xiàn)類,擴(kuò)展如果依賴了其它的擴(kuò)展,做不到自動(dòng)注入和裝配,擴(kuò)展很難和其它框架集成。
也正是基于這種種原因,許多框架中不會(huì)去直接使用ServiceLoader這種原生的SPI機(jī)制而是會(huì)去基于這種思想進(jìn)行一定的擴(kuò)展,使其的功能更加強(qiáng)大,典型的案例就是dubbo的SPI,SpringBoot的SPI。
小編的這篇文章《SpringBoot借助spring.factories文件跨模塊實(shí)例化Bean》就是講SpringBoot中的SPI機(jī)制,感興趣的同學(xué)可以閱讀一下。
到此這篇關(guān)于Java中的SPI機(jī)制案例分享的文章就介紹到這了,更多相關(guān)Java中的SPI機(jī)制內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
spring boot+mybatis搭建一個(gè)后端restfull服務(wù)的實(shí)例詳解
這篇文章主要介紹了spring boot+mybatis搭建一個(gè)后端restfull服務(wù),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11
JAVA 文件監(jiān)控 WatchService的示例方法
本篇文章主要介紹了JAVA 文件監(jiān)控 WatchService的示例方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-10-10
SpringSecurity?默認(rèn)登錄認(rèn)證的實(shí)現(xiàn)原理解析
這篇文章主要介紹了SpringSecurity?默認(rèn)登錄認(rèn)證的實(shí)現(xiàn)原理解析,本文通過圖文示例相結(jié)合給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-12-12
java面試突擊之sleep和wait有什么區(qū)別詳析
按理來說sleep和wait本身就是八竿子打不著的兩個(gè)東西,但是在實(shí)際使用中大家都喜歡拿他們來做比較,或許是因?yàn)樗鼈兌伎梢宰尵€程處于阻塞狀態(tài),這篇文章主要給大家介紹了關(guān)于java面試突擊之sleep和wait有什么區(qū)別的相關(guān)資料,需要的朋友可以參考下2022-02-02

