在.NET Core中批量註入Grpc服務

来源:https://www.cnblogs.com/ZaraNet/archive/2020/01/08/12167517.html
-Advertisement-
Play Games

GRPC 是谷歌發佈的一個開源、高性能、通用RPC服務,儘管大部分 RPC 框架都使用 TCP 協議,但其實 UDP 也可以,而 gRPC 乾脆就用了 HTTP2。還有就是它具有跨平臺、跨語言 等特性,這裡就不再說明RPC是啥。 在寫項目當中,grp服務過多會非常頭疼,那麼我們分析一下如果解決這個問 ...


  GRPC 是谷歌發佈的一個開源、高性能、通用RPC服務,儘管大部分 RPC 框架都使用 TCP 協議,但其實 UDP 也可以,而 gRPC 乾脆就用了 HTTP2。還有就是它具有跨平臺、跨語言 等特性,這裡就不再說明RPC是啥。

  在寫項目當中,grp服務過多會非常頭疼,那麼我們分析一下如果解決這個問題。我們都知道在grpc註入到.NET Core 中使用的方法是 MapGrpcService 方法,是一個泛型方法。

    [NullableAttribute(0)]
    [NullableContextAttribute(1)]
    public static class GrpcEndpointRouteBuilderExtensions
    {
        public static GrpcServiceEndpointConventionBuilder MapGrpcService<TService>(this IEndpointRouteBuilder builder) where TService : class;
    }

那我們就可以通過反射調用這個方法來進行服務批量註冊,看方法的樣子我們只需要將我們的服務對應 TService 以及將我們的 endpointBuilder 傳入即可,我們看下源碼是不是就像我所說的那樣?

    public static class GrpcEndpointRouteBuilderExtensions
    {
        public static GrpcServiceEndpointConventionBuilder MapGrpcService<TService>(this IEndpointRouteBuilder builder) where TService : class
        {
            if (builder == null)
            {
                throw new ArgumentNullException(nameof(builder));
            }

            ValidateServicesRegistered(builder.ServiceProvider);

            var serviceRouteBuilder = builder.ServiceProvider.GetRequiredService<ServiceRouteBuilder<TService>>();
            var endpointConventionBuilders = serviceRouteBuilder.Build(builder);

            return new GrpcServiceEndpointConventionBuilder(endpointConventionBuilders);
        }

        private static void ValidateServicesRegistered(IServiceProvider serviceProvider)
        {
            var marker = serviceProvider.GetService(typeof(GrpcMarkerService));
            if (marker == null)
            {
                throw new InvalidOperationException("Unable to find the required services. Please add all the required services by calling " +
                    "'IServiceCollection.AddGrpc' inside the call to 'ConfigureServices(...)' in the application startup code.");
            }
        }
    }

  ok,看樣子沒什麼問題就像我剛纔所說的那樣做。現在我們準備一個proto以及一個Service.這個就在網上找個吧..首先定義一個proto,它是grpc中的協議,也就是每個消費者遵循的。

syntax = "proto3";
option csharp_namespace = "Grpc.Server";
package Greet;
service Greeter {
  rpc SayHello (HelloRequest) returns (HelloReply);
}
message HelloRequest {
  string name = 1;
  enum Laguage{
      en_us =0 ;
      zh_cn =1 ;
  }
  Laguage LaguageEnum = 2;
}
message HelloReply {
  string message = 1;
  int32 num = 2;
  int32 adsa =3;
}

隨後定義Service,當然非常簡單, Greeter.GreeterBase 是重新生成項目根據proto來生成的。

public class GreeterService : Greeter.GreeterBase
    {
        public override Task<HelloReply> SayHello(HelloRequest request, ServerCallContext context)
        {
            var greeting = string.Empty;
            switch (request.LaguageEnum)
            {
                case HelloRequest.Types.Laguage.EnUs:
                    greeting = "Hello";
                    break;
                case HelloRequest.Types.Laguage.ZhCn:
                    greeting = "你好";
                    break;
            }
            return Task.FromResult(new HelloReply
            {
                Message = $"{greeting} {request.Name}",
                Num = new Random().Next()
            });
        }
    }

此時我們需要自定義一個中間件,來批量註入grpc服務,其中我們獲取了類型為 GrpcEndpointRouteBuilderExtensions ,並獲取了它的方法,隨後傳入了他的TService,最後通過Invoke轉入了我們的終點對象。

public static class GrpcServiceExtension
    {
        public static void Add_Grpc_Services(IEndpointRouteBuilder builder)
        {
            Assembly assembly = Assembly.GetExecutingAssembly(); 
            foreach (var item in ServicesHelper.GetGrpcServices("Grpc.Server"))
            {
                Type mytype = assembly.GetType(item.Value + "."+item.Key);
                var method = typeof(GrpcEndpointRouteBuilderExtensions).GetMethod("MapGrpcService").MakeGenericMethod(mytype);
                method.Invoke(null, new[] { builder }); 
            };
        }
        public static void useMyGrpcServices(this IApplicationBuilder app)
        {
            app.UseEndpoints(endpoints =>
            {
                Add_Grpc_Services(endpoints);
            });
        }
    }

在 ServicesHelper 中通過反射找到程式集當中的所有文件然後判斷並返回。

 public static class ServicesHelper
    {
        public static Dictionary<string,string> GetGrpcServices(string assemblyName)
        {
            if (!string.IsNullOrEmpty(assemblyName))
            {
                Assembly assembly = Assembly.Load(assemblyName);
                List<Type> ts = assembly.GetTypes().ToList();

                var result = new Dictionary<string, string>();
                foreach (var item in ts.Where(u=>u.Namespace == "Grpc.Server.Services"))
                {
                    result.Add(item.Name,item.Namespace);
                }
                return result;
            }
            return new Dictionary<string, string>();
        }
}

這樣子我們就註入了所有命名空間為Grpc.Server.Services的服務,但這樣好像無法達到某些控制,我們應當如何處理呢,我建議攜程Attribute的形式,創建一個Flag.

public class GrpcServiceAttribute : Attribute
{
        public bool IsStart { get; set; }
}

將要在註入的服務商添加該標識,例如這樣。

 [GrpcService]
    public class GreeterService : Greeter.GreeterBase
    {...}    

隨後根據反射出來的值找到 AttributeType 的名稱進行判斷即可。

 public static Dictionary<string,string> GetGrpcServices(string assemblyName)
        {
            if (!string.IsNullOrEmpty(assemblyName))
            {
                Assembly assembly = Assembly.Load(assemblyName);
                List<Type> ts = assembly.GetTypes().ToList();

                var result = new Dictionary<string, string>();
                foreach (var item in ts.Where(u=>u.CustomAttributes.Any(a=>a.AttributeType.Name == "GrpcServiceAttribute")))
                {
                    result.Add(item.Name,item.Namespace);
                }
                return result;
            }
            return new Dictionary<string, string>();
        }

隨後我們的批量註入在Starup.cs中添加一行代碼即可。

app.useMyGrpcServices();

啟動項目試一試效果:

示例代碼:傳送門


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

-Advertisement-
Play Games
更多相關文章
  • 攔截器介紹 mybatis提供了 @Intercepts 註解允許開發者對mybatis的執行器Executor進行攔截。 Executor介面方法主要有update、query、commit、rollback等等。 主要思路為: 1. 進入攔截器方法中 2. 獲取攔截器方法參數 3. 獲取解析參數 ...
  • 1. Java中有幾種方法可以實現一個線程?用什麼關鍵字修飾同步方法?stop()和suspend()方法為何不推薦使用? 三種實現方法 1.繼承 Thread 類 擴展性差 無返回值 2, 實現 Runnable 介面 可擴展 無返回值 3. 實現 Callable 介面 有返回值 用synchr ...
  • 這篇文章主要是介紹生成器和IO多路復用機制, 算是學習asyncio需要的預備知識. 這個系列還有另外兩篇文章: 從零開始學asyncio(中) 從零開始學asyncio(下) 一. 簡單爬蟲實例 首先創建一個crawler.py文件, 寫入以下代碼: import socket req = 'GE ...
  • 目的:修改VS Code的註釋文本顏色 S1:假設VS Code的安裝路徑是 %MVSC% S2:文件資源管理器進入目錄 %MVSC%\resources\app\extensions\ S3:該目錄底下由若幹以“theme-”開頭的目錄,例如: theme-abyss theme-defaults ...
  • 控制台錯誤提示: 2020-01-08 18:34:40,292 [http-nio-8080-exec-3] [org.apache.struts2.dispatcher.Dispatcher]-[WARN] Could not find action or result: /views/dire ...
  • 背景 程式在發佈部署時候,設置環境 不生效,也沒在代碼里使用 ,啟動一直是 .最後測試發現只有在 中配置 才生效,網上找了半天資料也沒看到有什麼問題。 最終翻看源代碼,發現是在 中的 替換了全局 導致。 平時開發大體知道程式啟動時候埠啟用順序是 環境變數 預設,具體是怎麼確定使用哪個配置的,沒找到 ...
  • asp.net core 實現支持多語言 Intro 最近有一個外國友人通過郵件聯繫我,想用我的活動室預約,但是還沒支持多語言,基本上都是寫死的中文,所以最近想支持一下更多語言,於是有了多語言方面的一些實踐 國際化/本地化介紹 國際化(Globalization)和本地化(Localization) ...
  • 上篇我們說到。編寫控制器類的步驟可總結為兩個:實現一個類,然後在該類中添加一些公有方法,在運行的該類的時候可作為控制器發現,而這些方法則作為操作被髮現。 這裡我們有兩個細節: 1:系統如何知道實例化那個控制器 2:如何確定用那個方法。 路由: 1:被傳統的路由發現,2:通過特性路由發現,3:通過混合 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...