91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

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

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

什么是ServiceLoader解耦服務

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

本篇內容介紹了“什么是ServiceLoader解耦服務”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

ServiceLoader解耦服務

背景

SPI,全稱Service Provider Interfaces,服務提供接口。是Java提供的一套供第三方實現或擴展使用的技術體系。主要通過解耦服務具體實現以及服務使用,使得程序的可擴展性大大增強,甚至可插拔。

基于服務的注冊與發現機制,服務提供者向系統注冊服務,服務使用者通過查找發現服務,可以達到服務的提供與使用的分離,甚至完成對服務的管理。

JDK中,基于SPI的思想,提供了默認具體的實現,ServiceLoader。利用JDK自帶的ServiceLoader,可以輕松實現面向服務的注冊與發現,完成服務提供與使用的解耦

完成分離后的服務,使得服務提供方的修改或替換,不會給服務使用方帶來代碼上的修改,基于面向接口的服務約定,提供方和使用方各自直接面向接口編程,而不用關注對方的具體實現。同時,服務使用方使用到服務時,也才會真正意義上去發現服務,以完成服務的初始化,形成了服務的動態加載

原理

  • ServiceLoader代碼片段

  •  

  • 說明

外部使用時,往往通過load(Class<S> service, ClassLoader loader)load(Class<S> service)調用,最后都是在reload方法中創建了LazyIterator對象,LazyIteratorServiceLoader的內部類,實現了Iterator接口,其作用是一個懶加載的迭代器,在hasNextService方法中,完成了對位于META-INF/services/目錄下的配置文件的解析,并在nextService方法中,完成了對具體實現類的實例化。

JDK的ServiceLoader通過返回一個Iterator對象能夠做到對服務實例的懶加載 只有當調用iterator.next()方法時才會實例化下一個服務實例,只有需要使用的時候才進行實例化,具體實現讀者可以去閱讀源碼進行研究,這也是其設計的亮點之一。

META-INF/services/,是ServiceLoader中約定的接口與實現類的關系配置目錄,文件名是接口全限定類名,內容是接口對應的具體實現類,如果有多個實現類,分別將不同的實現類都分別作為每一行去配置。解析過程中,通過LinkedHashMap<String,S>數據結構的providers,將已經發現了的接口實現類進行了緩存,并對外提供的iterator()方法,方便外部遍歷。

總體上,ServiceLoader的一般實現與使用過程包含了服務接口約定服務實現服務注冊服務發現與使用這四個步驟。

應用

r2dbc-mariadb實際也是基于r2dbc-spi的方式實現服務解耦

  • 服務接口約定: r2dbc-spi

  • 服務實現:r2dbc-mariadb對r2dbc-spi的實現

  • 服務注冊:r2dbc-mariadb包中META-INF.service文件


  • 服務發現:在r2dbc-spi中創建連接工廠的方法

    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能夠發現ZenithConnectionFactoryProvider,是因為在r2dbc-zenith的META-INF.service文件中添加響應的類路徑聲明(注冊自己)以供ServerLoader掃描。

其他應用,如:jdbc-driver

“什么是ServiceLoader解耦服務”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

太保市| 专栏| 余庆县| 南平市| 兰考县| 上栗县| 临泉县| 濉溪县| 吉木乃县| 治多县| 襄樊市| 大城县| 忻城县| 漾濞| 呼伦贝尔市| 桂东县| 石渠县| 手机| 高阳县| 嘉善县| 武义县| 金寨县| 钟山县| 玛多县| 会理县| 新邵县| 龙口市| 濉溪县| 永靖县| 西充县| 方正县| 柳林县| 普格县| 巴林左旗| 阿勒泰市| 安顺市| 鸡东县| 娄底市| 容城县| 上杭县| 彰化县|