這篇文章主要為大家詳細介紹了SignalR Self Host+MVC等多端訊息推送服務,具有一定的參考價值,感興趣的小夥伴們可以參考一下
一、概述
由於項目需要,最近公司項目裡有個模組功能,需要使用到即時獲得審批通知;原本的設計方案是使用ajax對伺服器進行定時輪詢查詢,剛剛開始資料量和使用量不大的時候還好,後來使用量的增加和系統中各種業務的複雜度增加,伺服器的壓力也越來越大,於是我想使用訊息推送的方式替換掉ajax輪詢查詢,當有審批提交時,調用推送方法,將訊息推送到下一審批人那,這樣就減低了伺服器的壓力。
Signal 是微軟支援的一個運行在.NET平台上的 html websocket 架構。它出現的主要目的是實現伺服器主動推送訊息到用戶端頁面,這樣用戶端就不必重新發送請求或使用輪詢技術來擷取訊息。而且SignalR的相容性也是很強大的,這裡不在多言。既然選擇了SignalR,那麼就開始幹吧!
我的想法是將SignalR做成一個自託管的服務,和我們的b/s項目分離出來,這樣的好處是,1、推送服務不依賴於iis,就算iis掛了,我們的推送服務還可以正常運行;2、我們可以多平台叫用這個推送服務,多重專案都可以同時使用;
二、建立服務端
廢話不多說了,我也是第一次寫部落格,介紹完業務情境和構思,我們就開始擼碼吧。
1、用VS建立一個名為 "SignalRProject" 的解決方案;
2、在 SignalRProject解決方案下建立一個名為Server的控制台
3、在封裝管理員控制台,輸入如下命令
Install-Package Microsoft.AspNet.SignalR.SelfHost
4、輸入如下命令:
Install-Package Microsoft.Owin.Cors
5、在Server控制台中添加UserInfo類,代碼如下
using System; namespace Server { public class UserInfo { public string ConnectionId { get; set; } public string UserName { get; set; } public DateTime LastLoginTime { get; set; } } }
6、在Server控制台中添加ChatHub類,代碼如下
using Microsoft.AspNet.SignalR;using Microsoft.AspNet.SignalR.Hubs;using System;using System.Collections.Generic;using System.Linq;using System.Threading.Tasks;namespace Server{ [HubName("IMHub")] public class ChatHub : Hub { // 靜態屬性 public static List<UserInfo> OnlineUsers = new List<UserInfo>(); // 線上使用者列表 /// <summary> /// 登入連線 /// </summary> /// <param name="userId">使用者Id</param> /// <param name="userName">使用者名稱</param> public void Register(string userName) { var connnectId = Context.ConnectionId; if (OnlineUsers.Count(x => x.ConnectionId == connnectId) == 0) { if (OnlineUsers.Any(x => x.UserName == userName)) { var items = OnlineUsers.Where(x => x.UserName == userName).ToList(); foreach (var item in items) { Clients.AllExcept(connnectId).onUserDisconnected(item.ConnectionId, item.UserName); } OnlineUsers.RemoveAll(x => x.UserName == userName); } //添加線上人員 OnlineUsers.Add(new UserInfo { ConnectionId = connnectId, UserName = userName, LastLoginTime = DateTime.Now }); } // 所有用戶端同步線上使用者 Clients.All.onConnected(connnectId, userName, OnlineUsers); } /// <summary> /// 發送私聊 /// </summary> /// <param name="toUserId">接收方使用者串連ID</param> /// <param name="message">內容</param> public void SendPrivateMessage(string toUserName, string message) { var fromConnectionId = Context.ConnectionId; var toUser = OnlineUsers.FirstOrDefault(x => x.UserName == toUserName); var fromUser = OnlineUsers.FirstOrDefault(x => x.ConnectionId == fromConnectionId); if (toUser != null ) { Clients.Client(toUser.ConnectionId).receivePrivateMessage(fromUser.UserName, message); Clients.Client(toUser.ConnectionId).receivePrivateMessage(message); } else { //表示對方不線上 Clients.Caller.absentSubscriber(); } } public void Send(string name, string message) { //Clients.All { get; } // 代表所有用戶端 //Clients.AllExcept(params string[] excludeConnectionIds); // 除了參數中的所有用戶端 //Clients.Client(string connectionId); // 特定的用戶端,這個方法也就是我們實現端對端聊天的關鍵 //Clients.Clients(IList<string> connectionIds); // 參數中的用戶端 //Clients.Group(string groupName, params string[] excludeConnectionIds); // 指定用戶端組,這個也是實現群聊的關鍵所在 //Clients.Groups(IList<string> groupNames, params string[] excludeConnectionIds);參數中的用戶端組 //Clients.User(string userId); // 特定的使用者 //Clients.Users(IList<string> userIds); // 參數中的使用者 Console.WriteLine("ConnectionId:{0}, InvokeMethod:{1}", Context.ConnectionId, "Send"); Clients.All.addMessage(name, message); } /// <summary> /// 連線時調用 /// </summary> /// <returns></returns> public override Task OnConnected() { Console.WriteLine("用戶端串連,串連ID是:{0},當前線上人數為{1}", Context.ConnectionId, OnlineUsers.Count+1); return base.OnConnected(); } /// <summary> /// 斷線時調用 /// </summary> /// <param name="stopCalled"></param> /// <returns></returns> public override Task OnDisconnected(bool stopCalled) { var user = OnlineUsers.FirstOrDefault(u => u.ConnectionId == Context.ConnectionId); // 判斷使用者是否存在,存在則刪除 if (user == null) { return base.OnDisconnected(stopCalled); } Clients.All.onUserDisconnected(user.ConnectionId, user.UserName); //調用用戶端使用者離線通知 // 刪除使用者 OnlineUsers.Remove(user); Console.WriteLine("用戶端斷線,串連ID是:{0},當前線上人數為{1}", Context.ConnectionId, OnlineUsers.Count); return base.OnDisconnected(stopCalled); } public override Task OnReconnected() { return base.OnReconnected(); } }}
7、在Server控制台中添加Startup類,代碼如下
using Microsoft.Owin.Cors;using Owin;namespace Server{ public class Startup { public void Configuration(IAppBuilder app) { //允許CORS跨域 app.UseCors(CorsOptions.AllowAll); app.MapSignalR(); } }}
8、修改Server控制台中添加Program類,代碼如下
using Microsoft.Owin.Hosting;using System;namespace Server{ class Program { static void Main(string[] args) { string url = "http://localhost:10086";//設定 SignalR Hub Server 對外的介面 using (WebApp.Start(url))//啟動 SignalR Hub Server { Console.WriteLine("Server running on {0}", url); Console.ReadLine(); } } }}
9、F5運行起來
然後瀏覽器中訪問http://localhost:10086/signalr/hubs
結果如下:
見內容就基本完成了,今天先講到著,時間不早了,先休息了,後續有時間再將後面的文章補上。