您好,登錄后才能下訂單哦!
本篇內(nèi)容介紹了“什么是ServiceLoader解耦服務(wù)”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!
SPI,全稱Service Provider Interfaces
,服務(wù)提供接口。是Java提供的一套供第三方實現(xiàn)或擴展使用的技術(shù)體系。主要通過解耦服務(wù)具體實現(xiàn)以及服務(wù)使用,使得程序的可擴展性大大增強,甚至可插拔。
基于服務(wù)的注冊與發(fā)現(xiàn)機制,服務(wù)提供者向系統(tǒng)注冊服務(wù),服務(wù)使用者通過查找發(fā)現(xiàn)服務(wù),可以達到服務(wù)的提供與使用的分離,甚至完成對服務(wù)的管理。
JDK中,基于SPI的思想,提供了默認(rèn)具體的實現(xiàn),ServiceLoader
。利用JDK自帶的ServiceLoader
,可以輕松實現(xiàn)面向服務(wù)的注冊與發(fā)現(xiàn)
,完成服務(wù)提供與使用的解耦
。
完成分離后的服務(wù),使得服務(wù)提供方的修改或替換,不會給服務(wù)使用方帶來代碼上的修改,基于面向接口的服務(wù)約定,提供方和使用方各自直接面向接口編程,而不用關(guān)注對方的具體實現(xiàn)。同時,服務(wù)使用方使用到服務(wù)時,也才會真正意義上去發(fā)現(xiàn)服務(wù)
,以完成服務(wù)的初始化
,形成了服務(wù)的動態(tài)加載
。
ServiceLoader代碼片段
說明
外部使用時,往往通過load(Class<S> service, ClassLoader loader)
或load(Class<S> service)
調(diào)用,最后都是在reload
方法中創(chuàng)建了LazyIterator
對象,LazyIterator
是ServiceLoader
的內(nèi)部類,實現(xiàn)了Iterator
接口,其作用是一個懶加載的迭代器,在hasNextService
方法中,完成了對位于META-INF/services/
目錄下的配置文件的解析,并在nextService
方法中,完成了對具體實現(xiàn)類的實例化。
JDK的ServiceLoader通過返回一個Iterator對象能夠做到對服務(wù)實例的懶加載 只有當(dāng)調(diào)用iterator.next()方法時才會實例化下一個服務(wù)實例,只有需要使用的時候才進行實例化,具體實現(xiàn)讀者可以去閱讀源碼進行研究,這也是其設(shè)計的亮點之一。
META-INF/services/
,是ServiceLoader
中約定的接口與實現(xiàn)類的關(guān)系配置目錄,文件名是接口全限定類名,內(nèi)容是接口對應(yīng)的具體實現(xiàn)類,如果有多個實現(xiàn)類,分別將不同的實現(xiàn)類都分別作為每一行去配置。解析過程中,通過LinkedHashMap<String,S>
數(shù)據(jù)結(jié)構(gòu)的providers
,將已經(jīng)發(fā)現(xiàn)了的接口實現(xiàn)類進行了緩存,并對外提供的iterator()
方法,方便外部遍歷。
總體上,ServiceLoader
的一般實現(xiàn)與使用過程包含了服務(wù)接口約定
、服務(wù)實現(xiàn)
、服務(wù)注冊
、服務(wù)發(fā)現(xiàn)與使用
這四個步驟。
r2dbc-mariadb實際也是基于r2dbc-spi的方式實現(xiàn)服務(wù)解耦
服務(wù)接口約定: r2dbc-spi
服務(wù)實現(xiàn):r2dbc-mariadb對r2dbc-spi的實現(xiàn)
服務(wù)注冊:r2dbc-mariadb包中META-INF.service文件
服務(wù)發(fā)現(xiàn):在r2dbc-spi中創(chuàng)建連接工廠的方法
public static ConnectionFactory find(ConnectionFactoryOptions connectionFactoryOptions) { Assert.requireNonNull(connectionFactoryOptions, "connectionFactoryOptions must not be null"); for (ConnectionFactoryProvider provider : loadProviders()) { if (provider.supports(connectionFactoryOptions)) { return provider.create(connectionFactoryOptions); } } return null; } private static ServiceLoader<ConnectionFactoryProvider> loadProviders() { return AccessController.doPrivileged((PrivilegedAction<ServiceLoader<ConnectionFactoryProvider>>) () -> ServiceLoader.load(ConnectionFactoryProvider.class, ConnectionFactoryProvider.class.getClassLoader())); }
之所以ServiceLoader能夠發(fā)現(xiàn)ZenithConnectionFactoryProvider,是因為在r2dbc-zenith的META-INF.service文件中添加響應(yīng)的類路徑聲明(注冊自己)以供ServerLoader掃描。
其他應(yīng)用,如:jdbc-driver
“什么是ServiceLoader解耦服務(wù)”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。