dubbo 的 spi 思想是什麼?

来源:https://www.cnblogs.com/midoujava/archive/2019/09/16/11530933.html
-Advertisement-
Play Games

面試題 dubbo 的 spi 思想是什麼? 面試官心理分析 繼續深入問唄,前面一些基礎性的東西問完了,確定你應該都 ok,瞭解 dubbo 的一些基本東西,那麼問個稍微難一點點的問題,就是 spi,先問問你 spi 是啥?然後問問你 dubbo 的 spi 是怎麼實現的? 其實就是看看你對 dub ...


面試題

dubbo 的 spi 思想是什麼?

面試官心理分析

繼續深入問唄,前面一些基礎性的東西問完了,確定你應該都 ok,瞭解 dubbo 的一些基本東西,那麼問個稍微難一點點的問題,就是 spi,先問問你 spi 是啥?然後問問你 dubbo 的 spi 是怎麼實現的?

其實就是看看你對 dubbo 的掌握如何。

面試題剖析

spi 是啥?

spi,簡單來說,就是 service provider interface,說白了是什麼意思呢,比如你有個介面,現在這個介面有 3 個實現類,那麼在系統運行的時候對這個介面到底選擇哪個實現類呢?這就需要 spi 了,需要根據指定的配置或者是預設的配置,去找到對應的實現類載入進來,然後用這個實現類的實例對象。

舉個慄子。

你有一個介面 A。A1/A2/A3 分別是介面A的不同實現。你通過配置 介面 A = 實現 A2,那麼在系統實際運行的時候,會載入你的配置,用實現 A2 實例化一個對象來提供服務。

spi 機制一般用在哪兒?插件擴展的場景,比如說你開發了一個給別人使用的開源框架,如果你想讓別人自己寫個插件,插到你的開源框架裡面,從而擴展某個功能,這個時候 spi 思想就用上了。

Java spi 思想的體現

spi 經典的思想體現,大家平時都在用,比如說 jdbc。

Java 定義了一套 jdbc 的介面,但是 Java 並沒有提供 jdbc 的實現類。

但是實際上項目跑的時候,要使用 jdbc 介面的哪些實現類呢?一般來說,我們要根據自己使用的資料庫,比如 mysql,你就將 mysql-jdbc-connector.jar 引入進來;oracle,你就將 oracle-jdbc-connector.jar 引入進來。

在系統跑的時候,碰到你使用 jdbc 的介面,他會在底層使用你引入的那個 jar 中提供的實現類。

dubbo 的 spi 思想

dubbo 也用了 spi 思想,不過沒有用 jdk 的 spi 機制,是自己實現的一套 spi 機制。

Protocol protocol = ExtensionLoader.getExtensionLoader(Protocol.class).getAdaptiveExtension();

Protocol 介面,在系統運行的時候,,dubbo 會判斷一下應該選用這個 Protocol 介面的哪個實現類來實例化對象來使用。

它會去找一個你配置的 Protocol,將你配置的 Protocol 實現類,載入到 jvm 中來,然後實例化對象,就用你的那個 Protocol 實現類就可以了。

上面那行代碼就是 dubbo 里大量使用的,就是對很多組件,都是保留一個介面和多個實現,然後在系統運行的時候動態根據配置去找到對應的實現類。如果你沒配置,那就走預設的實現好了,沒問題。

@SPI("dubbo")  
public interface Protocol {  
      
    int getDefaultPort();  
  
    @Adaptive  
    <T> Exporter<T> export(Invoker<T> invoker) throws RpcException;  
  
    @Adaptive  
    <T> Invoker<T> refer(Class<T> type, URL url) throws RpcException;  

    void destroy();  
  
}  

在 dubbo 自己的 jar 里,在/META_INF/dubbo/internal/com.alibaba.dubbo.rpc.Protocol文件中:

dubbo=com.alibaba.dubbo.rpc.protocol.dubbo.DubboProtocol
http=com.alibaba.dubbo.rpc.protocol.http.HttpProtocol
hessian=com.alibaba.dubbo.rpc.protocol.hessian.HessianProtocol

所以說,這就看到了 dubbo 的 spi 機制預設是怎麼玩兒的了,其實就是 Protocol 介面,@SPI("dubbo") 說的是,通過 SPI 機制來提供實現類,實現類是通過 dubbo 作為預設 key 去配置文件里找到的,配置文件名稱與介面全限定名一樣的,通過 dubbo 作為 key 可以找到預設的實現類就是 com.alibaba.dubbo.rpc.protocol.dubbo.DubboProtocol

如果想要動態替換掉預設的實現類,需要使用 @Adaptive 介面,Protocol 介面中,有兩個方法加了 @Adaptive 註解,就是說那倆介面會被代理實現。

啥意思呢?

比如這個 Protocol 介面搞了倆 @Adaptive 註解標註了方法,在運行的時候會針對 Protocol 生成代理類,這個代理類的那倆方法裡面會有代理代碼,代理代碼會在運行的時候動態根據 url 中的 protocol 來獲取那個 key,預設是 dubbo,你也可以自己指定,你如果指定了別的 key,那麼就會獲取別的實現類的實例了。

如何自己擴展 dubbo 中的組件

下麵來說說怎麼來自己擴展 dubbo 中的組件。

自己寫個工程,要是那種可以打成 jar 包的,裡面的 src/main/resources 目錄下,搞一個 META-INF/services,裡面放個文件叫:com.alibaba.dubbo.rpc.Protocol,文件里搞一個my=com.bingo.MyProtocol。自己把 jar 弄到 nexus 私服里去。

然後自己搞一個 dubbo provider 工程,在這個工程裡面依賴你自己搞的那個 jar,然後在 spring 配置文件里給個配置:

<dubbo:protocol name=”my” port=”20000” />

provider 啟動的時候,就會載入到我們 jar 包里的my=com.bingo.MyProtocol 這行配置里,接著會根據你的配置使用你定義好的 MyProtocol 了,這個就是簡單說明一下,你通過上述方式,可以替換掉大量的 dubbo 內部的組件,就是扔個你自己的 jar 包,然後配置一下即可。

file

dubbo 裡面提供了大量的類似上面的擴展點,就是說,你如果要擴展一個東西,只要自己寫個 jar,讓你的 consumer 或者是 provider 工程,依賴你的那個 jar,在你的 jar 里指定目錄下配置好介面名稱對應的文件,裡面通過 key=實現類

然後對於對應的組件,類似 <dubbo:protocol> 用你的那個 key 對應的實現類來實現某個介面,你可以自己去擴展 dubbo 的各種功能,提供你自己的實現。

本文在米兜公眾號鏈接:
https://mp.weixin.qq.com/s/91wA5v4B0usFHfnMSIk7nw

歡迎關註米兜Java,一個註在共用、交流的Java學習平臺。

file


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • FreeMarker 介紹 Apache FreeMarker™是一個 模板引擎 :一個Java庫,用於根據模板和更改數據生成文本輸出(HTML網頁,電子郵件,配置文件,源代碼等)。模板是用FreeMarker模板語言(FTL)編寫的,這是一種簡單的專用語言(不像PHP這樣的完整編程語言)。通常,使 ...
  • 最好有其他web框架基礎,不推薦小白閱讀, 1.視圖函數 1.1視圖函數的返回值 django要求視圖函數必須返回一個HTTPResponse對象,render函數和 redirect 函數其實都是返回了一個特定的HTTPResponse對象 1.2 reder函數 render函數用於渲染一個模板 ...
  • 使用工具:IronPython 工具介紹:是一種在 .NET 及 Mono上的 Python 實現,是一個開源的項目,基於微軟的 DLR 引擎。(個人理解就是在 .net上面運行Python代碼) 使用方法:先新建一個控制台應用程式 => 使用Nuget 添加IronPython包 => 在Main ...
  • 1、元素的分類 需求:有如下集合[11,22,33,44,55,66,77,88,99,90……],將所有大於66的值保存在字典的第一個key中,將小於66的值保存在第二個key的值中 代碼實現: 1 #定義一個list列表 2 li = [11,22,33,44,55,66,77,88,99,90 ...
  • 2019-09-16-23:09:06 自學Python的第六天,也是寫博客的第六天 今天學的內容是有關dict字典的用法 看視頻加上練習,目前還沒遇到有難點,但是感覺很不好的樣子 沒有難點以後突然出現一個有關字典的程式感覺要炸,還是得繼續掌握 看最後的代碼吧,有更好的請告訴我 我 是 一 條 快 ...
  • 一:MonoDB的簡單介紹 MongoDB是一個介於關係型資料庫與非關係型資料庫中間的資料庫,是使用C++進行編寫的,他的優點是在支持的查詢格式特別的強大,可以進行存儲比較複雜的數據類型,支持建立索引 二:下載 官方地址:https://www.mongodb.com/ 本教程下載 3.4版本:ht ...
  • Python分散式爬蟲必學框架Scrapy打造搜索引擎 部分課程截圖: 點擊鏈接或搜索QQ號直接加群獲取其它資料: 鏈接:https://pan.baidu.com/s/1-wHr4dTAxfd51Mj9DxiJ4Q 提取碼:ik1n 免費分享,如若鏈接失效請加群 其它資源在群里,私聊管理員即可免費 ...
  • 又是兩個月的時間過去了,上一次寫博客是7月14號,時間還是過的很快的,那麼問題來了,為什麼這麼長時間都沒有寫東西了呢?難道是在打醬油? 哈哈,說起來很慚愧,剛剛開始工作,碰到各種的問題要去學習要去解決,然後業餘的時間又去學了一些奇奇怪怪的東西,導致博客一直都落下了,歸根到底,還是自己懶惰了,因為心中 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...