Asp.Net Core使用SignalR進行服務間調用

来源:https://www.cnblogs.com/fanshaoO/archive/2018/12/21/10155735.html
-Advertisement-
Play Games

網上查詢過很多關於ASP.NET core使用SignalR的簡單例子,但是大部分都是簡易聊天功能,今天心血來潮就搞了個使用SignalR進行服務間調用的簡單DEMO。 至於SignalR是什麼我就不多說了,微軟官方文檔也不少。 第一步新建項目 所有VS開發第一步都是新建一個解決方案哈,這裡我就不多 ...


網上查詢過很多關於ASP.NET core使用SignalR的簡單例子,但是大部分都是簡易聊天功能,今天心血來潮就搞了個使用SignalR進行服務間調用的簡單DEMO。

至於SignalR是什麼我就不多說了,微軟官方文檔也不少。


第一步新建項目

所有VS開發第一步都是新建一個解決方案哈,這裡我就不多介紹如何新建項目啦~~

  • 開發環境,VS2017,.NET CORE 2.1
  • 新建兩個asp.net core項目

     

    如此簡單的操作大家都懂的

註入SignalR

在被調用的服務端的Startup.cs中註入SignalR 在asp.net core2.1中已經預設包含了SignalR的庫,直接在ConfigureServices方法中添加如下代碼:

services.AddSignalR().AddMessagePackProtocol()

AddMessagePackProtocol()表示啟用 MessagePack 支持在伺服器上

新建一個繼承Hub類的類

public class ServerHub : Hub
{
    
}

裡面可以實現需要被客戶端執行的方法。

配置Hub連接URL

Startup.csConfigure添加下麵代碼

    app.UseSignalR(routes =>
    {
        routes.MapHub<ServerHub>("/myserver");
    });

添加服務

新建一個Services文件夾,添加我們的服務介面和介面實現類。

 


 

public interface IMyService
{
    Task<string> SayHello();
    Task Sleep();
}
public class MyService : IMyService
{
    public async Task<string> SayHello()
    {
        return await Task.Factory.StartNew(() => "Hello");
    }

    public async Task Sleep()
    {
        await Task.Delay(3000);            
    }
}

在Startup.cs中的ConfigureServices方法中進行依賴註入,註意需要在AddSignalR()之前註入。

services.AddScoped<IMyService, MyService>();

在ServerHub添加依賴註入

在ServerHub中添加如下代碼

  private readonly IServiceProvider _serviceProvider;
    public ServerHub(IServiceProvider serviceProvider)
    {
        _serviceProvider = serviceProvider;
    }

因為我們需要使用到依賴註入獲取指定Service,所以只註入IServiceProvider,使用IServiceProvider動態獲取服務介面。

使用反射動態獲取服務介面並執行指定方法

為了可以動態的選擇服務並且執行服務相應的方法,我們使用反射來動態獲取。 這裡我們添加兩個方法,一個有返回值,一個沒有返回值

    public async Task<dynamic> Excute(string serverName,string functionName,object[] parameters)
    {
        return await Task.Factory.StartNew(() =>
        {
            var type = Type.GetType(serverName);
            var service = _serviceProvider.GetRequiredService(type);
            var method = type.GetMethod(functionName);
            var resultTask = method.Invoke(service, new object[] { }) as Task;
            dynamic result = resultTask.GetType().GetProperty("Result").GetValue(resultTask, null);
            return result;
        });
    }

    public async Task ExcuteWithoutResult(string serverName, string functionName, object[] parameters)
    {
        var type = Type.GetType(serverName);
        var service = _serviceProvider.GetRequiredService(type);
        var method = type.GetMethod(functionName);
        var resultTask = method.Invoke(service, new object[] { }) as Task;
        await resultTask;
        var msg = "task done";
        await Clients.Caller.SendAsync("callback",msg);
    }

方法中傳入三個參數

  • serverName:服務介面名稱(完整的命名空間)
  • functionName:方法的名稱
  • parameters: 方法所需的參數

  1. 使用Type.GetType(serverName)獲取服務介面的Type。
  2. 使用_serviceProvider.GetRequiredService(type)從依賴註入中獲取對應服務。
  3. 使用type.GetMethod(functionName)獲取需要執行的方法。
  4. 執行方法method.Invoke(service, new object[] { })

由於我們的服務是非同步方法,method.Invoke()返回的是一個object對象,所以在將method.Invoke()返回類型轉為Task類型。

resultTask.GetType().GetProperty("Result").GetValue(resultTask, null)是通過反射獲取TaskResult屬性來回去相應的返回結果。

因為我們不知道方法返回的Task中返回的結果類型是什麼,所以我們依舊使用反射來獲取Task的結果並使用dynamic接收。


無返回值的我們使用Clients.Caller.SendAsync()在任務處理結束後給調用方返回一條消息。


在調用方添加一個SingalRClient類

    public class SignalRClient
    {
        private readonly HubConnection connection;
        public SignalRClient()
        {
            connection = new HubConnectionBuilder()
                   .WithUrl("http://localhost:5000/myserver")
                   .AddMessagePackProtocol()
                   .Build();

            connection.Closed += async (error) =>
            {
                await Task.Delay(new Random().Next(0, 5) * 1000);
                await connection.StartAsync();
            };
            InitOnMethod();
            connection.StartAsync().ConfigureAwait(false).GetAwaiter().GetResult();
        }
    }

  

在構造函數中初始化SignalR連接

WithUrl("http://localhost:5000/myserver")是連接被調用方的URL

AddMessagePackProtocol()是使用快速和精簡的二進位序列化格式進行傳輸。 在connection.Closed加入連接關閉事件,關閉後自動重新連接。 InitOnMethod初始化服務方回調的監聽事件

    private void InitOnMethod()
    {
        connection.On<string>("callback",(msg)=> {
            Console.WriteLine($"------------{msg}----------");
        });
    }

connection.StartAsync()啟動連接。

添加兩個請求服務端的方法

一個有返回值,一個無返回值。

    public async Task<dynamic> RequestServer1()
    {
        var result = await connection.InvokeAsync<dynamic>("Excute", "SignalRServer1.Services.IMyService", "SayHello",new object[] { });
        
        return result;
    }

    public async Task RequestWithoutResult()
    {
        await connection.SendAsync("ExcuteWithoutResult", "SignalRServer1.Services.IMyService", "Sleep", new object[] { });
    }

需要返回值的我們使用connection.InvokeAsync()方法

不需要返回值的我們使用connection.SendAsync()方法

將SignalRClient以單例形式註冊依賴註入

Startup.cs中的ConfigureServices方法中添加services.AddSingleton<SignalRClient>()

使用SignalRClient請求服務

在控制器中將SignalRClient註入

 

    private readonly SignalRClient _signalRClient;

    public ValuesController(SignalRClient signalRClient)
    {
        _signalRClient = signalRClient;
    }
    // GET api/values
    [HttpGet]
    public async Task<ActionResult<IEnumerable<string>>> Get()
    {
        var str = await _signalRClient.RequestServer1().ConfigureAwait(false);
        await _signalRClient.RequestWithoutResult().ConfigureAwait(false);
        return new string[] { str };
    }

在請求中同時調用一個有返回值,一個無返回值的方法。無返回值的方法在任務執行完後執行一個回調。


啟動服務

 


 


 

 

 

可以看到服務調用已經成功 task done是我們無返回值調用那個方法中接收到回調時的輸出.

 

connection.On<string>("callback",(msg)=> {
            Console.WriteLine($"------------{msg}----------");
        });

  

 

結束啦~

以上就是我簡單做的一個DEMO。如有不足之處,請多多指點。基本沒寫過文章,寫的不好請諒解哈哈~~~

 


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

-Advertisement-
Play Games
更多相關文章
  • 1. 類的約束 1. 寫一個父類. 父類中的某個方法要拋出一個異常 NotImplementedError (重點) 2. 抽象類和抽象方法 2. 異常處理. 3. MD5 ...
  • 一、前言 flask中有很多可擴展點(筆者這樣稱呼),其中包含了信號和請求鉤子,這些信號和鉤子有什麼用呢?其主要作用用於幫助我們進行程式的耦合性,當然還可以讓我們自定義一些行為。話不多說,通過閱讀源碼,筆者將這些所有的可擴展點的執行順序進行總結(如下圖),這樣我們更能清楚的知道flask的內部請求流 ...
  • #第一個程式print('SigAi')#python 輸入程式name = input('please enter a name')print(name)#list 是可變有序集合#list之可變L = list(['a','b','c'])print(L)L.append('def')print... ...
  • 最近想做個全文搜索,設想用 ASP.NET Web API + Elasticsearch 6.x 來實現。 網上搜了下 Elasticsearch 的資料,大部分是講 linux 平臺下如何用 java 來開發,有少量講在 windows 平臺下用 c# 開發的,且版本是 Elasticsearc ...
  • 線上報來一個問題,說用戶的數據丟失了。開發經過緊張的調查。終於找到了原因。 問題出在了 if 中的比較上。 values[rowIndex] 中保存的是一個整數,開發認為兩個整數比較實用 == 就可以了。 但是 values[rowIndex] 中的整數經過 GetValue返回後被作為 objec ...
  • 既然標題為後續,就要放一下上一篇文章 "使用.Net Core 2.1開發Captcha圖片驗證碼服務" 繼續挖坑 時隔7個月再次繼續自己在GitHub上挖的坑 "https://github.com/PuzzledAlien/Captcha" 在之前翻譯了一片有關如何在.net core中使用Sy ...
  • 方法調用: ...
  • 如何在ADO中使用介面的抽象數據提供程式 在cofig中 appSettings下,配置數據連接類型 使用介面的抽象數據提供程式 ADO.NET數據提供程式工廠模型 .NET數據提供程式工廠模式能讓我們用多種數據訪問類型構建單個代碼庫。而且通過應用程式配置文件(和全新的<connectionstri ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...