溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

什么是ServiceLoader解耦服務(wù)

發(fā)布時間:2021-06-28 17:22:23 來源:億速云 閱讀:160 作者:chen 欄目:編程語言

本篇內(nèi)容介紹了“什么是ServiceLoader解耦服務(wù)”的有關(guān)知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

ServiceLoader解耦服務(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對象,LazyIteratorServiceLoader的內(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)與使用這四個步驟。

應(yīng)用

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ì)量的實用文章!

向AI問一下細(xì)節(jié)

免責(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)容。

AI