WPF+ASP.NET SignalR實現後臺通知

来源:https://www.cnblogs.com/hsiang/archive/2022/09/14/16687365.html
-Advertisement-
Play Games

在實際業務中,當後臺數據發生變化,客戶端能夠實時的收到通知,而不是由用戶主動的進行頁面刷新才能查看,這將是一個非常人性化的設計 ...


在實際業務中,當後臺數據發生變化,客戶端能夠實時的收到通知,而不是由用戶主動的進行頁面刷新才能查看,這將是一個非常人性化的設計。比如數字化大屏,並沒有人工的干預,而是自動的刷新數據,那如何才能實現數據的實時刷新呢?本文以一個簡單示例,簡述如何通過WPF+ASP.NET SignalR實現消息後臺通知以及數據的實時刷新,僅供學習分享使用,如有不足之處,還請指正。

 

 通過上一篇文章的學習,瞭解瞭如何通過SignalR實現線上聊天功能,在示例中,我們發現每一次的客戶端連接都是一個新的實例對象,所以沒有辦法在中心對象中存儲狀態信息,所以為了存儲用戶列表,我們採用了靜態變數的方式。並且線上聊天功能是用戶發送一條消息(Chat),然後觸發中心對象(ChatHub),轉發給另一個用戶(SendAsync)。那麼如果實現數字化大屏,需要服務端持續的往客戶端發送消息,而不是客戶端主動觸發,應該怎麼做呢?這就是本文需要分享的內容。

涉及知識點

在本示例中,涉及知識點如下所示:

  1. 開發工具:Visual Studio 2022 目標框架:.NET6.0
  2. ASP.NET SignalR,一個ASP .NET 下的類庫,可以在ASP .NET 的Web項目中實現實時通信,目前新版已支持.NET6.0及以上版本。在本示例中,作為消息通知的服務端。
  3. WPF,是微軟推出的基於Windows 的用戶界面框架,主要用於開發客戶端程式。

前提條件

實現服務端持續往客戶端發送消息,除了業務上的需求外,還需要滿足兩個條件:

  1. 在服務端有一個常駐記憶體對象,監聽數據變化。
  2. 常駐記憶體對象,可以訪問中心對象(ChatHub),能夠獲取中心對象的所有連接客戶端,併發送消息。

滿足以上兩個條件,才可以實現想要的功能。

服務端

經過以上分析後,服務端分為兩方面,核心對象(ChatHub),處理業務對象(Worker)。下麵我們逐一說明:

ChatHub 中心是用於向連接到 SignalR 伺服器的客戶端發送消息的核心抽象,負責客戶端的連接和斷開。如下所示:

 1 using Microsoft.AspNetCore.SignalR;
 2 
 3 namespace SignalRChat.Chat
 4 {
 5     public class ChatHub:Hub
 6     {
 7         public override Task OnConnectedAsync()
 8         {
 9             Console.WriteLine($"ID:{Context.ConnectionId} 已連接");
10             return base.OnConnectedAsync();
11         }
12 
13         public override Task OnDisconnectedAsync(Exception? exception)
14         {
15             Console.WriteLine($"ID:{Context.ConnectionId} 已斷開");
16             return base.OnDisconnectedAsync(exception);
17         }
18     }
19 }

Worker實例為一個單例對象,常駐內容,實時監聽數據變化,並通過ChatHub上下文(IHubContext<ChatHub>)獲取連接信息,然後發送消息,如下所示:

 1 using Microsoft.AspNetCore.SignalR;
 2 
 3 namespace SignalRChat.Chat
 4 {
 5     public class Worker
 6     {
 7         public static Worker Instance;
 8         
 9         private static readonly object locker=new object();
10 
11         private IHubContext<ChatHub> context;
12 
13         private System.Timers.Timer timer;
14 
15         public Worker(IHubContext<ChatHub> context) { 
16             this.context = context;
17             timer= new System.Timers.Timer(500);//單位毫秒
18             timer.Enabled=true;
19             timer.AutoReset=true;//自動重新
20             timer.Elapsed += Timer_Elapsed;
21             timer.Start();
22         }
23 
24         private void Timer_Elapsed(object? sender, System.Timers.ElapsedEventArgs e)
25         {
26             //模擬數據,一般情況下,從資料庫獲取,然後通知到客戶端
27             Dictionary<string, object> data = new Dictionary<string, object>();
28             var online = new Random().Next(0, 100);
29             var male = Math.Floor(new Random().NextSingle() * online);
30             var female = online - male;
31             data["online"]=online;
32             data["male"] =male;
33             data["female"] = female;
34             context.Clients.All.SendAsync("Data",data);
35         }
36 
37         public static void Register(IHubContext<ChatHub> context)
38         {
39             if (Instance == null)
40             {
41                 lock (locker)
42                 {
43                     if (Instance == null)
44                     {
45                         Instance = new Worker(context);
46                     }
47                 }
48             }
49         }
50     }
51 }

註意:此處發送數據的是Data方法,客戶端必須監聽Data方法,才能接收數據。

如何創建單例對象呢,中心對象上下文不能自己創建,必須要和ChatHub通過註入方式的上下文是同一個,不然無法獲取客戶端連接信息。在項目啟動時,通過中間件的方式創建,如下所示:

 1 using Microsoft.AspNetCore.SignalR;
 2 using SignalRChat.Chat;
 3 
 4 var builder = WebApplication.CreateBuilder(args);
 5 
 6 // Add services to the container.
 7 
 8 builder.Services.AddControllers();
 9 // Learn more about configuring Swagger/OpenAPI at https://aka.ms/aspnetcore/swashbuckle
10 builder.Services.AddEndpointsApiExplorer();
11 builder.Services.AddSwaggerGen();
12 //1.添加SignalR服務
13 builder.Services.AddSignalR();
14 var app = builder.Build();
15 
16 // Configure the HTTP request pipeline.
17 if (app.Environment.IsDevelopment())
18 {
19     app.UseSwagger();
20     app.UseSwaggerUI();
21 }
22 app.UseRouting();
23 app.UseHttpsRedirection();
24 
25 app.UseAuthorization();
26 //在Use中註冊單例實例
27 app.Use(async (context, next) =>
28 {
29     var hubContext = context.RequestServices
30                             .GetRequiredService<IHubContext<ChatHub>>();
31     Worker.Register(hubContext);//調用靜態方法註冊
32 
33     if (next != null)
34     {
35         await next.Invoke();
36     }
37 });
38 app.MapControllers();
39 //2.映射路由
40 app.UseEndpoints(endpoints => {
41     endpoints.MapHub<ChatHub>("/chat");
42 });
43 
44 app.Run();

客戶端

客戶端主要是連接伺服器,然後監聽服務端發送數據的方法即可,如下所示:

  1 namespace SignalRClient
  2 {
  3     public class ShowDataViewModel : ObservableObject
  4     {
  5         #region 屬性及構造函數
  6 
  7         private int online;
  8 
  9         public int Online
 10         {
 11             get { return online; }
 12             set { SetProperty(ref online, value); }
 13         }
 14 
 15         private int male;
 16 
 17         public int Male
 18         {
 19             get { return male; }
 20             set { SetProperty(ref male, value); }
 21         }
 22 
 23 
 24         private int female;
 25 
 26         public int Female
 27         {
 28             get { return female; }
 29             set { SetProperty(ref female, value); }
 30         }
 31 
 32         private HubConnection hubConnection;
 33 
 34         public ShowDataViewModel()
 35         {
 36 
 37         }
 38 
 39         #endregion
 40 
 41         #region 命令
 42 
 43         private ICommand loadedCommand;
 44 
 45         public ICommand LoadedCommand
 46         {
 47             get
 48             {
 49                 if (loadedCommand == null)
 50                 {
 51                     loadedCommand = new RelayCommand<object>(Loaded);
 52                 }
 53                 return loadedCommand;
 54             }
 55         }
 56 
 57         private void Loaded(object obj)
 58         {
 59             //1.初始化
 60             InitInfo();
 61             //2.監聽
 62             Listen();
 63             //3.連接
 64             Link();
 65         }
 66 
 67         #endregion
 68 
 69         /// <summary>
 70         /// 初始化Connection對象
 71         /// </summary>
 72         private void InitInfo()
 73         {
 74             hubConnection = new HubConnectionBuilder().WithUrl("https://localhost:7149/chat").WithAutomaticReconnect().Build();
 75             hubConnection.KeepAliveInterval = TimeSpan.FromSeconds(5);
 76         }
 77 
 78         /// <summary>
 79         /// 監聽
 80         /// </summary>
 81         private void Listen()
 82         {
 83             hubConnection.On<Dictionary<string,object>>("Data", ReceiveInfos);
 84         }
 85 
 86         /// <summary>
 87         /// 連接
 88         /// </summary>
 89         private async void Link()
 90         {
 91             try
 92             {
 93                 await hubConnection.StartAsync();
 94             }
 95             catch (Exception ex)
 96             {
 97                 MessageBox.Show(ex.Message);
 98             }
 99         }
100 
101         private void ReceiveInfos(Dictionary<string, object> data)
102         {
103             if (data == null || data.Count < 1)
104             {
105                 return;
106             }
107             int.TryParse(data["online"]?.ToString(),out int online);
108             int.TryParse(data["male"]?.ToString(),out int male);
109             int.TryParse(data["female"]?.ToString(),out int female);
110             this.Online=online;
111             this.Male = male;
112             this.Female=female;
113         }
114     }
115 }

註意:監聽Data方法,和服務端發送時保持一致。

運行示例

在示例中,需要同時啟動服務端和客戶端,所以以多項目方式啟動,如下所示:

 

 運行成功後,服務端以ASP.NET Web API的方式呈現,如下所示:

 

 客戶端運行如下:

註意:客戶端可以有多個,也可以是一個,後臺通知消息,會通知到每一個連接的客戶端。

源碼下載

關註微信公眾號,然後發送信息SignalR即可獲取下載鏈接。如下所示:

 

 

 

備註

以上就是WPF+ASP.NET SignalR實現後臺實時通知的全部內容,關於SignalR的應用,實際場景有很多,這隻是一個簡單的入門示例,希望可以拋磚引玉,一起學習,共同進步。學習編程,從關註【老碼識途】開始!!!


作者:小六公子
出處:http://www.cnblogs.com/hsiang/
本文版權歸作者和博客園共有,寫文不易,支持原創,歡迎轉載【點贊】,轉載請保留此段聲明,且在文章頁面明顯位置給出原文連接,謝謝。
關註個人公眾號,定時同步更新技術及職場文章


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

-Advertisement-
Play Games
更多相關文章
  • 可用三種不同的方式將 const 關鍵字用於一級指針,如下所示: //方式一:指向常量數據的指針,以下幾種為等效表示 const int * ptc; //方式一 int const * ptc; //方式二 //方式二:指針本身為常量,需在聲明時初始化 int x = 55; int * cons ...
  • 前言 嗨嘍,大家好呀~這裡是愛看美女的茜茜吶 又到了學Python時刻~ 今天實現一下人臉識別。 先問大家一個問題什麼是百度Aip模塊? 百度AI平臺提供了很多的API介面供開發者快速的調用運用在項目中本文寫的是使用百度AI的線上介面SDK模塊(baidu-aip)進行實現人臉識別 除了人臉識別,其 ...
  • 本文詳細闡述了knn演算法,從開始介紹什麼事knn,到講解knn演算法的原理再到最後以實際例子來運用knn演算法的步驟,實際例子的代碼講解也十分詳細 ...
  • 為了在jupyter中使用pyTorch的虛擬環境,來記錄一下怎麼操作一、conda命令的使用因為使用的是jupyter,所有就使用Anaconda Prompt來創建虛擬環境(也可使用virtualenv,不過沒試過) conda create -n 環境名 # 創建的環境在預設路徑下,C盤位置不 ...
  • 摘要:本文主要講述如何進行圖像量化處理和採樣處理及局部馬賽克特效。 本文分享自華為雲社區《[Python圖像處理] 二十.圖像量化處理和採樣處理及局部馬賽克特效》,作者: eastmount。 本文主要講述如何進行圖像量化處理和採樣處理及局部馬賽克特效。 一.圖像量化處理 圖像通常是自然界景物的客觀 ...
  • 1.推導式套路 除了最簡單的列表推導式和生成器表達式,其實還有字典推導式、集合推導式等等。 下麵是一個以列表推導式為例的推導式詳細格式,同樣適用於其他推導式。 variable = [out_exp_res for out_exp in input_list if out_exp == 2] out ...
  • 當面試官問你,“什麼是令牌桶限流演算法”! 你知道要怎麼回答,才能獲得面試官的青睞嗎? 大家好,我是Mic,一個工作了14年的Java程式員。 關於這個問題,面試官想考察哪些緯度?我們又該怎麼回答呢? 問題解析 限流策略,是在高併發流量下保護系統穩定性的一種策略。 所以這個問題,主要是互聯網公司會去考 ...
  • 一、argparse簡介 argparse 是 python 自帶的命令行參數解析包,可以用來方便的服務命令行參數,使用之前需要先導入包 import argparse 二、簡單案例 簡單使用,創建一個名為test.py的文件 # 導入 argparse 模塊 import argparse # 創 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...