SignalR的通信方式决定了其高性能,可是即使如此,当消息的并发量上来之后,单节点的Hub服务器依然可能没法承载总的消息吞吐量,那么如何对Hub服务器作水平扩展呢?服务器
从微软官方的文档上看,SignalR是具备消息底板功能的,SignalR核心组件公开了一个IMessageBus的接口,只须要实现该接口,就能实现消息订阅功能。官网提供了3种解决方案:Azure、Redis、SqlServer,nuget平台上有更多的基于消息队列的第三方底板。本篇以Redis为例子展现一下以消息底板模式运做的Hub服务器。并发
服务端app
public class Startup { public void Configuration(IAppBuilder app) { GlobalHost.DependencyResolver.UseRedis("192.168.1.66", 6379, string.Empty, "SignalRBus"); app.Map("/signalr", map => { map.UseCors(CorsOptions.AllowAll); var hubConfiguration = new HubConfiguration { EnableJSONP = true }; map.RunSignalR(hubConfiguration); }); } }
在启动类中,经过DependencyResolver的扩展方法来注册基于Redis的MessageBus,其中前2个参数是地址和端口,第三个参数为Redis服务器的密码,消息订阅的名称。性能
public class ChatHub : Hub { public void Chat(string msg) { Clients.All.Display("Receive Msg:" + msg); } }
创建一个简单的Hub,只提供一个Chat的方法,内部实现广播。ui
static void Main(string[] args) { using (WebApp.Start<Startup>("http://*:8001/")) { Console.WriteLine("Server running at http://localhost:8001/"); Console.ReadLine(); } }
static void Main(string[] args) { using (WebApp.Start<Startup>("http://*:8002/")) { Console.WriteLine("Server running at http://localhost:8002/"); Console.ReadLine(); } }
创建2个启动项,端口号分别为8001和8002,用以启动2个hub的实例。spa
客户端code
static void Main(string[] args) { var hubConn = new HubConnection("http://localhost:8001"); var proxy = hubConn.CreateHubProxy("ChatHub"); hubConn.Start().Wait(); while (true) { var guid = Guid.NewGuid().ToString(); Console.WriteLine("Send Msg:" + guid); proxy.Invoke("Chat", guid).Wait(); Thread.Sleep(2000); } }
客户端1链接上8001,利用一个循环不断的往服务端发guid。blog
static void Main(string[] args) { var hubConn = new HubConnection("http://localhost:8002"); var proxy = hubConn.CreateHubProxy("ChatHub"); proxy.On<string>("Display", (msg) => { Console.WriteLine(msg); }); hubConn.Start().Wait(); Console.ReadLine(); }
客户端2链接上8002,注册一个Display的方法,用于接收服务端推送的消息。接口
运行状况队列
图中可知,client1的send msg,被client2接受到了。