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


    一、概述

    由于项目需要,最近公司项目里有个模块功能,需要使用到即时获得审批通知;原本的设计方案是使用ajax对服务器进行定时轮询查询,刚刚开始数据量和使用量不大的时候还好,后来使用量的增加和系统中各种业务的复杂度增加,服务器的压力也越来越大,于是我想使用消息推送的方式替换掉ajax轮询查询,当有审批提交时,调用推送方法,将消息推送到下一审批人那,这样就减低了服务器的压力。

    SignalR是微软支持的一个运行在.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<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();
            }
        }
    }
    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<Startup>(url))//启动 SignalR Hub Server
                {
                    Console.WriteLine("Server running on {0}", url);
                    Console.ReadLine();
                }
            }
        }
    }
    View Code

     

    9、F5运行起来

    然后浏览器中访问http://localhost:10086/signalr/hubs

    结果如下:

    见上图内容就基本完成了,今天先讲到着,时间不早了,先休息了,后续有时间再将后面的文章补上

  • 相关阅读:
    Mysql的四种引擎介绍
    电商项目实战(架构三)——redis的启动,安装和运用(windows下)
    bug篇——ERROR: Maven JVM terminated unexpectedly with exit code 137
    电商项目实战(架构二)——Swagger-UI实现在线api文档
    电商项目实战(架构一)——SpringBoot+MyBatis搭建基本骨架
    bug篇——上传图片大小限制配置
    bug篇——Mybatis中Mapper.xml文件的if判断问题
    bug篇——线上移动端跳转页面
    安装篇——xshell评估过期解决
    bug篇——vue跨域时cookies问题
  • 原文地址:https://www.cnblogs.com/landonzeng/p/landon.html
Copyright © 2020-2023  润新知