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
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...