簡易RPC框架-SPI

来源:https://www.cnblogs.com/ASPNET2008/archive/2018/05/20/9062341.html
-Advertisement-
Play Games

案例 我們所熟悉的jbdc是一種用於執行SQL語句的Java API,可以為多種關係資料庫提供統一訪問,提供了一種基準,據此可以構建更高級的工具和介面。 如上圖所示,任意的一個資料庫廠商只要去實現jdbc的介面,就可以輕鬆的對接jbdc從而為應用開發人員所服務。 SPI 上面的jdbc的設計理念叫S ...


案例

我們所熟悉的jbdc是一種用於執行SQL語句的Java API,可以為多種關係資料庫提供統一訪問,提供了一種基準,據此可以構建更高級的工具和介面。

如上圖所示,任意的一個資料庫廠商只要去實現jdbc的介面,就可以輕鬆的對接jbdc從而為應用開發人員所服務。

SPI

上面的jdbc的設計理念叫SPI,它的全名是Service Provider Interface。它的理念是對某類功能進行抽象,確保應用程式依賴抽象而不是具體的某種實現,通過配置服務實現者的方式來達到面向介面編程以及擴展的目的。比如我們項目中需要用到日誌組件,有的項目喜歡logback,有的喜歡log4j,有的喜歡common-log等等,如果在項目中直接依賴這些日誌介面,那麼後續如果需要對日誌組件重新選型,對現有項目的影響會非常大,所有後來就有了slfj,它抽象了日誌介面但不包含任何的實現,具體實現全部依賴於不同的廠商。

傳統的java spi一般做法是在resources/META-INF/services/目錄下麵創建一個以服務介面命名的文件,該文件內容就是實現該服務介面的具體實現類的完全限定名。當程式載入的時候,就能通過resources/META-INF/services/里的配置文件找到具體的實現類名,並載入實例化。 通過這個機制就能找到服務介面的實現類,而不需要再代碼里寫死。

java.util.ServiceLoader這個就是java spi中用來載入服務實現類的工具,本文不對它的具體用法做過多介紹。

主題:限流策略如何擴展

本文要討論的問題是,rpc框架中的限流過濾器擴展問題(可參考之前的文章 :簡易RPC框架-客戶端限流配置),之前介紹的限流實現是採用了guava提供的RateLimit,當時客戶端限流的實現是在框架中寫好的不允許修改,不同項目如果需要不同的限流策略那麼就需要針對原有方案進行擴展,如果擴展呢?

Spring-boot 中的SPI

我在spring-boot項目中按傳統的spi方式配置後,發現ServiceLoader載入指定介面找不到具體的實現類,後來發現spring-boot有自己的spi實現。它是在resources/META-INF/spring.factories中配置相關的介面,而且這個類的配置方式與傳統的spi也有所不同,它採用了key=value方式,這點有點類似dubbo的spi機制。文件目錄如下:

下麵給出我調整之後的方案:

客戶端限流介面

定義一個限流的介面,因為限流會有些參數控制,所以就增加RpcInvocation來協助完成。

public interface AccessLimitService {

    void acquire(RpcInvocation invocation);
}

客戶端限流介面實現

本文只是為了簡單實現,所以直接將原有寫在rpc框架中的限流方式抽取出來,並沒有重新採用一種新的限流策略。

public class AccessLimitServiceImpl implements AccessLimitService {

    @Override
    public void acquire(RpcInvocation invocation) {
        AccessLimitManager.acquire(invocation);
    }


    static class AccessLimitManager{

        private final static Object lock=new Object();

        private final static Map<String,RateLimiter> rateLimiterMap= Maps.newHashMap();

        public static void acquire(RpcInvocation invocation){
            if(!rateLimiterMap.containsKey(invocation.getClassName())) {
                synchronized (lock) {
                    if(!rateLimiterMap.containsKey(invocation.getClassName())) {
                        final RateLimiter rateLimiter = RateLimiter.create(invocation.getMaxExecutesCount());
                        rateLimiterMap.put(invocation.getClassName(), rateLimiter);
                    }
                }
            }
            else {
                RateLimiter rateLimiter=rateLimiterMap.get(invocation.getClassName());
                rateLimiter.acquire();
            }
        }
    }
}

客戶端限流過濾器調整

既然限流的實現抽取成了介面,所以此處的具體實現調整為從服務提供者中找對應的實現。

@Override
public Object invoke(RpcInvoker invoker, RpcInvocation invocation) {
    logger.info("before acquire,"+new Date());
    List<AccessLimitService> accessLimitServiceLoader = SpringFactoriesLoader.loadFactories(AccessLimitService.class, null);
    if(!CollectionUtils.isEmpty(accessLimitServiceLoader)){
        AccessLimitService accessLimitService=accessLimitServiceLoader.get(0);
        accessLimitService.acquire(invocation);
    }

    Object rpcResponse=invoker.invoke(invocation);
    logger.info("after acquire,"+new Date());
    return rpcResponse;
}
    

目前還不支持同一個項目中多種限流策略,目前版本只允許存在一種,如果配置了多種實現,也只會選擇第一個。如果需要支持也是可以的,通過配置一個名稱來指定即可,但感覺價值並不大。

SpringFactoriesLoader就是spring-boot實現的類似java.util.ServiceLoader的一種服務載入工具,它負責從resources/META-INF/spring.factories中讀取相應的配置,並對其載入實例化。總共包含兩個核心方法:

  • loadFactoryNames

    這個方法就是載入某個介面的所有指定實現類名,它可以服務於下麵的loadFactories方法。

  • loadFactories 首先通過loadFactoryNames方法從配置文件中獲取介面與實現類的關係,然後一個一個實例化服務實現類。

經過以上幾步的調整,就基本實現了一個簡單的基於SPI思想的組件擴展機制。客戶端可以擴展任意的限流機制去替換。

本文源碼

文中代碼是依賴上述項目的,如果有不明白的可下載源碼


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

-Advertisement-
Play Games
更多相關文章
  • NSSortDescriptor 排序 Person類 1 #import <Foundation/Foundation.h> 2 3 @interface Person : NSObject 4 5 @property (nonatomic, copy) NSString *name; 6 @pr ...
  • 一、todolist功能開發 <div id="root"> <div> <input type="text" v-model="inputValue"> <button @click="handleSubmit">提交</button> </div> <ul> <li v-for="(item, ...
  • 今天終於能更新博文了……(確實有些不應該,5月份到現在就沒認認真真寫過幾行代碼) 今天算是正式在學Vue.js了,前期想必大家也能看得出來,我為那個即時通訊系統寫的代碼註釋,基本上都是自己在胡亂猜測的,很隨意很隨性的在寫註釋(可能都完全不正確…) 今天簡單的抄寫了兩個官網上的例子,花了半個小時左右的 ...
  • 前言 我的目標是寫一個非常詳細的關於diff的乾貨,所以本文有點長。也會用到大量的圖片以及代碼舉例,目的讓看這篇文章的朋友一定弄明白diff的邊邊角角。 先來瞭解幾個點... 1. 當數據發生變化時,vue是怎麼更新節點的? 要知道渲染真實DOM的開銷是很大的,比如有時候我們修改了某個數據,如果直接 ...
  • 深入css佈局(1)—— 盒模型 & 元素分類 “ 在css知識體系中,除了css選擇器,樣式屬性等基礎知識外,css佈局相關的知識才是css比較核心和重要的點。今天我們來深入學習一下css佈局相關的知識。” 首先來列下大綱 盒模型 IE盒模型 W3C盒模型 box-sizing 元素的分類 塊級元 ...
  • 談談對事件的理解: 從前有一家人,我們稱為window家。window他其中有一個兒子叫事件。 事件 事件他有很多好基友,每次發生什麼事情都會找到事件,叫他幫忙查看一下那些細節出現了問題。可是事件這個小子有時候遇到同一類型的事情會忘記前面那個小伙伴的問題,只記得最後的那個問題。但有時荷爾蒙分泌過多的 ...
  • console.log('請輸入三個數:'); let num1 = readline.question() - 0; let num2 = readline.question() - 0; let num3 = readline.question() - 0; let num4; if (num1 ...
  • 最近做個一個功能需要用到自動補全,然後在網上找了很久,踩了各種的坑 最後用typeahead.js這個插件,經過自己的測試完美實現 使用方法:在頁面中引入jquery、jquery.typeahead.min.js和jquery.typeahead.min.css文件。 html部分代碼: jque ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...