SignalR Self Host+MVC等多端消息推送服務(1)

来源:http://www.cnblogs.com/landonzeng/archive/2017/06/27/landon.html
-Advertisement-
Play Games

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 
View Code

4、輸入如下命令: 

 Install-Package Microsoft.Owin.Cors
View Code

 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; }
    }
}

View Code

 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 OnlineUsers = new List(); // 線上用戶列表

        ///
        /// 登錄連線
        ///
        ///用戶Id
        ///用戶名
        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);
        }

        ///
        /// 發送私聊
        ///
        ///接收方用戶連接ID
        ///內容
        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 connectionIds); // 參數中的客戶端
            //Clients.Group(string groupName, params string[] excludeConnectionIds); // 指定客戶端組,這個也是實現群聊的關鍵所在
            //Clients.Groups(IList groupNames, params string[] excludeConnectionIds);參數中的客戶端組
            //Clients.User(string userId);  // 特定的用戶
            //Clients.Users(IList userIds); // 參數中的用戶

            Console.WriteLine("ConnectionId:{0}, InvokeMethod:{1}", Context.ConnectionId, "Send");
            Clients.All.addMessage(name, message);
        }

        ///
        /// 連線時調用
        ///
        ///
        public override Task OnConnected()
        {
            Console.WriteLine("客戶端連接,連接ID是:{0},當前線上人數為{1}", Context.ConnectionId, OnlineUsers.Count+1);
            return base.OnConnected();
        }


        ///
        /// 斷線時調用
        ///
        ///
        /// 
        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();
        }
    }
}

View Code

 

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();
        }
    }
}

View Code

 

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();
            }
        }
    }
}
View Code

 

9、F5運行起來

然後瀏覽器中訪問http://localhost:10086/signalr/hubs

結果如下:

見上圖內容就基本完成了,今天先講到著,時間不早了,先休息了,後續有時間再將後面的文章補上


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

-Advertisement-
Play Games
更多相關文章
  • 系統管理賬戶登錄系統會大大降低系統的安全性,所以為了更安全的使用電腦,通常我們都會創建一個新的用戶來登錄系統,MySQL也不例外。MySQL中創建用戶與授權的實現方法,對於剛開始接觸mysql的朋友可以參考下! 創建 Mysql創建用戶的方法分成三種:INSERTUSER表的方法、CREATEUS ...
  • 1.查詢dblink語句 col owner for a20col db_link for a30col username for a20col host for a30set linesize 120set pages 60select * from dba_db_links order by o ...
  • 官方文檔鏈接 http://docs.oracle.com/cd/E11882_01/server.112/e41134/protection.htm SBYDB02000 最大可用模式(Maximum Availability) 這種保護模式在 不影響主庫可用性 的前提下提供最高水平的數據保護 在 ...
  • 第一次寫博客,哈哈,內心有種無法掩蓋的小激動。 我有個問題,如何才能成為一名好程式員?是代碼風騷?還是會不間斷的學習?歡迎大家給出自己的建議。 好了,廢話不多說了,進入今天的正題。 我不知道你們在用windows版的redis的時候有沒有遇到這麼幾個坑: 1.# Warning: no config ...
  • 存儲數據是為了查找數據,存儲結構影響數據查找的性能。對無序數據進行查找,最快的查找演算法是哈希查找;對有序數據進行查找,最快的查找演算法是平衡樹查找。在傳統的關係型資料庫中,聚集索引和非聚集索引都是平衡樹(B-Tree)類型的存儲結構,用於順序存儲數據,便於實現數據的快速查找。除了提升數據查找的性能之外 ...
  • " 1、Oracle 中的三大分頁方法 " "1.1、通過分析函數分頁" "1.2、通過 ROWNUM 分頁" "1.3、通過 ROWID 分頁" " 2、Oracle 分頁解決方案淺析 " "2.1、純後端代碼完成分頁" "2.2、通過存儲過程來分頁" "2.3、兩個通用的分頁存儲過程" " 3、 ...
  • 本文介紹hdfs的存儲策略 內容譯自:http://hadoop.apache.org/docs/r2.8.0/hadoop-project-dist/hadoop-hdfs/ArchivalStorage.html#Get_Storage_Policy 譯註:存儲策略是在和經濟妥協的一個技術產物。 ...
  • [20170625]12c Extended statistics.txt--//別人的系統12c,awr報表出現大量調用執行如下sql語句.select default$ from col$ where rowid=:1;--//google看了一下,問題出在Extended statistics ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...