Toggle navigation
首页
产品中心
全新RDIF.vNext低代码框架
镇店
.NET C/S开发框架
.NET Web敏捷开发框架
.NET 快速开发框架(全新EasyUI版本)
.NET 代码生成器
.NET WebAPI服务端开发框架
客户案例
付款方式
国思学堂
技术文章
新闻资讯
下载
关于
首页
技术文章
.NET(Core)
正文
原创
2019-08-21
浏览 (
13804
)
史上最全面的SignalR系列教程-4、SignalR 自托管全解(使用Self-Host)-附各终端详细实例
## 1、概述 ## 通过前面几篇文章 [RDIFramework.NET敏捷开发框架通过SignalR技术整合即时通讯(IM)](http://www.guosisoft.com/article/detail/229) [史上最全面的SignalR系列教程-1、认识SignalR](http://www.guosisoft.com/article/detail/225) [史上最全面的SignalR系列教程-2、SignalR 实现推送功能-永久连接类实现方式](http://www.guosisoft.com/article/detail/226) [史上最全面的SignalR系列教程-3、SignalR 实现推送功能-集线器类实现方式](http://www.guosisoft.com/article/detail/227) 我们对SignalR的概念以及SignalR的最主要的两类通信模型(Persistent Connections与Hubs)进行了详细的对比讲解,也做了案例展示。本篇将为大家介绍.NET特有的Self-Host自托管的应用,即以Self-Host自托管为宿主加载SignalR服务。 宿主一词我们不会陌生,它可以看作是一个基础设施,它为一些服务和功能提供最底层的支持,如你的web应用程序可以运行在iis或者apache上,而这两个东西就是web应用程序的宿主,而今天说的自主宿主SelfHost它可以自己去监听自己的服务,如你可以把一个web应用程序宿主到一个console控制台程序上,或者把一个webApi宿主到一个console或者windowService上,这都是可以的。 Self-Host的介绍我们可以参考msdn官方事例[https://docs.microsoft.com/en-us/dotnet/framework/wcf/samples/self-host](https://docs.microsoft.com/en-us/dotnet/framework/wcf/samples/self-host) 这是一个wcf的事例。 SignalR常常依托于ASP.NET应用程序运行于IIS中,但它还可以自我托管(比如作为console winform、Windows service),只要我们使用self-host库就可以了。该库向所有的SignalR 2库一样,构建于OWIN (Open Web Interface for .NET)。OWIN定义了一个在.NET web 服务端和web 应用程序的抽象。OWIN解耦了从服务端来的web 应用程序,这使得OWIN对于子托管web应用程序于自己的进程中得以表现得很完美。 有的小伙伴可能就要问了,为什么一定要使用这种方式来托管呢?基于IIS的方式不是更简单吗?不托管于IIS的原因主要有以下方面的考虑: - 我们已经构建了一个服务端程序,没有运行在IIS上,重新搭建成本太高。 - IIS的方式性能开销很大。 - signalR需要扩展到现有程序中,如Windows服务中等。 - 其他的一些原因.... 本篇主要讲解以下内容: 1. 使用Self-Host方式创建SignalR服务端 2. 用javascript客户端访问SignalR Self-Host服务端 3. 用WinForm客户端接收SignalR Self-Host服务端消息 ## 2、使用Self-Host方式创建SignalR服务端 ## 为了更简单的说明self-host托管的方式,我们用控制台应用程序来做演示。当然我们也可以自托管到Windows服务中去,如果你更希望托管到Windows服务,可以参考[Self-Hosting SignalR in a Windows Service](http://code.msdn.microsoft.com/SignalR-self-hosted-in-6ff7e6c3) 在上一项目的基础上,我们新建一个名为:SignalRSelfHost的控制台应用程序,如下所所示: ![新建SignalRSelfHost控制台应用程序](http://doc.rdiframework.net/blog/article/20190820094153518.png-pw) 要使用Selft-Host宿主SignalR,必须引用Microsoft.AspNet.SignalR.SelfHost包。我们在程序包管理控制台输入以下命令安装SelfHost包。 Install-Package Microsoft.AspNet.SignalR.SelfHost 要想支持跨域访问,还需要安装 Install-Package Microsoft.Owin.Cors 如果用命令的方式安装失败,我们可以用NuGet的方式安装,如下图所示。 ![NuGet方式安装Self-Host](http://doc.rdiframework.net/blog/article/20190820094328500.png-pw) ![NuGet方式安装Self-Host](http://doc.rdiframework.net/blog/article/20190820094349260.png-pw) 编写代码如下: using Microsoft.AspNet.SignalR; using Microsoft.Owin.Cors; using Microsoft.Owin.Hosting; using Owin; using System; namespace SignalRSelfHost { class Program { static void Main(string[] args) { // This will *ONLY* bind to localhost, if you want to bind to all addresses // use http://*:8080 to bind to all addresses. // See http://msdn.microsoft.com/en-us/library/system.net.httplistener.aspx // for more information. string url = "http://localhost:8077"; using (WebApp.Start(url)) { Console.WriteLine("SignalR Server running on {0}", url); Console.ReadLine(); } } } /// <summary> /// 该类含有SignalR服务端的配置(该教程使用的唯一的配置是用来调用UseCors), /// MapSignalR为所有形式的Hub对象创建了路由规则 /// </summary> class Startup { public void Configuration(IAppBuilder app) { app.UseCors(CorsOptions.AllowAll); app.MapSignalR(); } } /// <summary> /// SignalR的Hub 类是程序要提供给客户端的 /// 该类就一个方法,Send:客户端可以用来发送消息给其他客户端 /// </summary> public class MyHub : Hub { //服务端的方法,客户端可以去调用 public void Send(string name, string message) { //调用客户端的方法addMessage(string s1,string s2); Clients.All.addMessage(name, message); } } } **代码说明:** - Program:包含程序的主方法。在这个方法中,类型为Startup的web应用程序启动于指定的URL (http://localhost:8077)。 如果需要更加安全一点,可以支持SSL,请去这里看看How to: [Configure a Port with an SSL Certificate](http://msdn.microsoft.com/en-us/library/ms733791.aspx) - Startup: 该类含有SignalR服务端的配置(该教程使用的唯一的配置是用来调用UseCors), MapSignalR为所有形式的Hub对象创建了路由规则。 - MyHub:SignalR的Hub 类是程序要提供给客户端的。 该类就一个方法:Send, 客户端可以用来发送消息给其他客户端。 编译运行SignalR控制台服务端如下。 ![SignalR控制台服务端启动](http://doc.rdiframework.net/blog/article/20190820094421023.png-pw) 可以看到我们的控制台服务端已经成功启动起来了。 ## 3、用javascript客户端访问SignalR Self-Host服务端 ## ### 3.1、新建SignalRSelfHostJSClient的MVC空项目 ### ![新建SignalRSelfHostJSClient的MVC空项目](http://doc.rdiframework.net/blog/article/20190820094457067.png-pw) ### 3.2、添加Microsoft.AspNet.SignalR.JS的引用 ### ![添加Microsoft.AspNet.SignalR.JS的引用](http://doc.rdiframework.net/blog/article/20190820094525814.png-pw) ### 3.3、新建Default.html页面 ### 代码如下: <!DOCTYPE html> <html> <head> <title>SignalRSelfHostJSClient Chat</title> <style type="text/css"> .container { background-color: #a1c6ec; border: thick solid #e62fc7; padding: 20px; margin: 20px; } </style> </head> <body> <div class="container"> <input type="text" id="message" style="width:350px;" placeholder="请输入消息内容..."/> <input type="button" id="sendmessage" value="发送" /> <input type="hidden" id="displayname" /> <ul id="discussion"></ul> </div> <!--Script references. --> <!--Reference the jQuery library. --> <script src="Scripts/jquery-1.6.4.min.js"></script> <!--Reference the SignalR library. --> <script src="Scripts/jquery.signalR-2.4.1.min.js"></script> <!--Reference the autogenerated SignalR hub script. --> <script src="http://localhost:8077/signalr/hubs"></script> <!--Add script to update the page and send messages.--> <script type="text/javascript"> $(function () { //Set the hubs URL for the connection $.connection.hub.url = "http://localhost:8077/signalr"; // Declare a proxy to reference the hub. var chat = $.connection.myHub; // Create a function that the hub can call to broadcast messages. chat.client.addMessage = function (name, message) { // Html encode display name and message. var encodedName = $('<div />').text(name).html(); var encodedMsg = $('<div />').text(message).html(); // Add the message to the page. $('#discussion').prepend('<li><strong>' + encodedName + '</strong>: ' + encodedMsg + '</li>'); }; // Get the user name and store it to prepend to messages. $('#displayname').val(prompt('Enter your name:', '')); // Set initial focus to message input box. $('#message').focus(); // Start the connection. $.connection.hub.start().done(function () { $('#sendmessage').click(function () { // Call the Send method on the hub. chat.server.send($('#displayname').val(), $('#message').val()); // Clear text box and reset focus for next comment. $('#message').val('').focus(); }); }); }); </script> </body> </html> ### 3.4、运行效果 ### 先运行SignalRSelfHost控制台服务端后,再运行两个我们的测试客户端,分别输入不同的用户名,试试聊天的效果,如下图所示。 ![SignalRSelfHostJSClient运行效果](http://doc.rdiframework.net/blog/article/20190820094629973.png-pw) ## 4、用WinForm客户端接收SignalR Self-Host服务端消息 ## ### 4.1、新建SignalRSelfHost的WinForm项目 ### ![新建SignalRSelfHost的WinForm项目](http://doc.rdiframework.net/blog/article/20190820094722464.png-pw) ### 4.2、使用NuGet安装Microsoft.AspNet.SignalR.Client包 ### ![NuGet安装Microsoft.AspNet.SignalR.Client包](http://doc.rdiframework.net/blog/article/20190820094805326.png-pw) ### 4.3、制作测试界面如下: ### ![SignalRWinForm测试客户端设计界面](http://doc.rdiframework.net/blog/article/20190820094831820.png-pw) 编写界面代码如下: using Microsoft.AspNet.SignalR.Client; using System; using System.Configuration; using System.Windows.Forms; namespace SignalRSelfHostWinFormClient { /// <summary> /// SignalRSelfHost WinForm测试客户端 /// 用于接收消息(需配合SignalRSelfHostJSClient项目使用) /// </summary> public partial class SignalRSelfHostWinFormClientTest : Form { //定义代理,广播服务连接相关 private static IHubProxy HubProxy { get; set; } private static readonly string ServerUrl = ConfigurationManager.AppSettings["SignalRServer"]; //定义一个连接对象 public static HubConnection Connection { get; set; } public SignalRSelfHostWinFormClientTest() { CheckForIllegalCrossThreadCalls = false; InitializeComponent(); } private void SignalRSelfHostWinFormClientTest_Load(object sender, EventArgs e) { Connection = new HubConnection(ServerUrl); Connection.Closed += Connection_Closed; HubProxy = Connection.CreateHubProxy("MyHub"); HubProxy.On<string, string>("addMessage", RecvMsg);//接收实时信息 Connection.Start().ContinueWith(task => { if (!task.IsFaulted) { msgContent.AppendText(string.Format("与Signal服务器连接成功,服务器地址:{0}\r\n",ServerUrl)); } else { msgContent.AppendText("与服务器连接失败,请确认服务器是否开启。\r\n"); } }).Wait(); } private void Connection_Closed() { msgContent.AppendText("连接关闭...\r\n"); } private void RecvMsg(string name, string message) { msgContent.AppendText(string.Format("接收时间:{0},发送人:{1},消息内容:{2},\r\n", DateTime.Now, name, message)); } } } ### 4.4、运行效果: ### ![SignalRWinForm测试客户端运行效果](http://doc.rdiframework.net/blog/article/20190820094901410.png-pw) 通过以上的详细讲解,我们已经非常清楚的了解了如何通过SignalR打通各终端以实现相互沟通的通信。例子虽然比较简洁,但完全可以以此为基础,扩展更复杂的业务应用。 ## 5、代码下载 ## 实例源码可以移步github下载,地址:[https://github.com/yonghu86/SignalRTestProj](https://github.com/yonghu86/SignalRTestProj) ## 6、参考文章 ## - [RDIFramework.NET敏捷开发框架通过SignalR技术整合即时通讯(IM)](http://www.guosisoft.com/article/detail/229) - [史上最全面的SignalR系列教程-1、认识SignalR](http://www.guosisoft.com/article/detail/225) - [史上最全面的SignalR系列教程-2、SignalR 实现推送功能-永久连接类实现方式](http://www.guosisoft.com/article/detail/226) - [史上最全面的SignalR系列教程-3、SignalR 实现推送功能-集线器类实现方式](http://www.guosisoft.com/article/detail/227) - [Real-time ASP.NET with SignalR](https://dotnet.microsoft.com/apps/aspnet/signalr) - [微信公众号开发系列-玩转微信开发-目录汇总](http://www.guosisoft.com/article/detail/216) - [RDIFramework.NET — 基于.NET的快速信息化系统开发框架 — 系列目录](http://www.guosisoft.com/article/detail/190) - [RDIFramework.NET ━ .NET快速信息化系统开发框架 ━ 工作流程组件介绍](http://www.guosisoft.com/article/detail/169) - [RDIFramework.NET框架SOA解决方案(集Windows服务、WinForm形式与IIS形式发布)-分布式应用](http://www.guosisoft.com/article/detail/189) - [RDIFramework.NET代码生成器全新V3.5版本发布-重大升级](http://www.guosisoft.com/article/detail/199) ----- 一路走来数个年头,感谢RDIFramework.NET框架的支持者与使用者,大家可以通过下面的地址了解详情。 RDIFramework.NET官方网站:http://www.rdiframework.net/ RDIFramework.NET官方博客:http://blog.rdiframework.net/ 同时需要说明的,以后的所有技术文章以官方网站为准,欢迎大家收藏! RDIFramework.NET框架由海南国思软件科技有限公司专业团队长期打造、一直在更新、一直在升级,请放心使用! 欢迎关注RDIFramework.net框架官方公众微信(微信号:guosisoft),及时了解最新动态。 扫描二维码立即关注 ![微信号:guosisoft](http://doc.rdiframework.net/weixin.png)
正文到此结束
本文标签:
.NET
挨踢业界
RDIF.NET
版权声明:
本站原创文章,由
guosisoft.com
发布,遵循
CC 4.0 by-sa
版权协议,转载请附上原文出处链接和本声明。
上一篇
史上最全面的SignalR系列教程-3、SignalR 实现推送功能-集线器类实现方式
下一篇
史上最全面的SignalR系列教程-5、SignalR 实现一对一聊天
热门推荐
{{article.title}}
热门指数:
浏览({{article.lookCount + 5000}})
相关文章
{{article.title}}
该篇文章的评论功能暂时被站长关闭
说给你听
本文目录
文章标签
RDIF.NET
其他
微信开发
.NET
消息交互
.NetCore
项目管理
常用工具
工作流
Web前端
数据库
挨踢业界
随机文章
.NET快速信息化系统开发框架 V3.2->新增“行政区域管理”,同时大批量树采用异步加载
RDIFramework.NET V3.3 Web版新增报表管理功能模块-重量级实用功能
RDIFramework.NET CS敏捷开发框架 V6.1发布(.NET6+、Framework双引擎、全网唯一)
企业数字化转型如何做?看过来
值得一看的35个Redis常用问题总结
《ORACLE PL/SQL编程详解》全原创(共八篇)--系列文章导航
微信公众号开发C#系列-12、微信前端开发利器:WeUI
.NET快速信息化系统开发框架 V3.2 新增解压缩工具类ZipHelper
国思RDIF.vNext全新低代码快速开发框架平台发布
后台界面设计之表格设计规范参考
RDIFramework.NET ━ .NET快速信息化系统开发框架 记录所有操作的Sql
RDIFramework.NET—.NET快速开发整合框架【开发实例】之产品管理(WinForm版)
.NET Core部署到linux(CentOS)最全解决方案,进阶篇(Supervisor+Nginx)
RDIFramework.NET开发框架WinForm版新增编码管理
后台界面设计之表单设计规范参考
ORACLE常见问题一千问(提供下载)(不怕学不成、就怕心不诚!)
.NET快速信息化系统开发框架 V3.2-新增锁定用户与解除锁定用户的功能
RDIFramework.NET开发框架在线表单设计助力可视化快速开发
常用linux命令,开发必备
RDIFramework.NET(.NET快速开发框架) 答客户问(2014-02-23)
网站信息
文章总数:599 篇
标签总数:8 个
分类总数:8 个
留言数量:1385 条
在线人数:
89
人
运行天数:1321天
最后更新:2023-05-18
QQ:406590790
13005007127