SignalR 聊天室实例详解(服务器端推送版)
翻译自:http://www.codeproject.com/Articles/562023/Asp-Net-SignalR-Chat-Room (在这里可以下载到实例的源码)
Asp.Net SignalR 聊天室
简单介绍:
最近微软向ASP.NET Web Application Framework中添加了很多的新特性,SignalR是其中比较显著的特性之一,它是用来创建实时的应用程序的,例如:社交应用(social application)、多人游戏、新闻天气等,在实时的应用程序中,一旦内容可以产生马上就会被推送给用户,它为远程呼叫(remote procedure calls)提供一个asp.net API,用来在服务器端通过.net代码调用一个客户端浏览器中的js方法。
web应用程序通过请求处理模型来工作,浏览器或者其他的代理发送一个请求,然后服务器端为请求提供相应在这个模型当中,服务器在没有被请去I的时候不能做出响应,然而在实时的应用程序服务器上,最新的可用内容会被推送到客户端你可以通过使用ASP.NET SignalR API 来达到这一特性
为了解释 SignalR Web API ,在这篇文章中,我会创建一个可以群组聊天也可以私聊的聊天程序,你需要用VS2012 来创建这个项目
你在这个网站可以获取更多的信息http://www.asp.net/signalr
让我们开始吧:
在VS中创建一个新的应用程序项目(web application project ),创建好了以后,右键——添加新项——选择SignalR Hub Class ,命名之后点击添加按钮,它会自动添加 Hub 类,需要的引用,还有脚本。

注册Hub URL
为了使用SignalR API,我们需要注册 ~/signalr/hubs URL,在你的解决方案中添加global.asax文件,在Application_Start方法中用RouteTable.Routes.MapHubs()来注册 Hub URL
public class Global : System.Web.HttpApplication
{
protected void Application_Start(object sender, EventArgs e)
{
// Register the default hubs route: ~/signalr/hubs
RouteTable.Routes.MapHubs();
}
}
Hub 类
ChatHub 类必须继承自Microsoft.AspNet.SignalR.Hub 类,Hub 类暴露出一些属性和方法,你可以通过Clients属性与保持链接的客户端交流
你可以得到正在调用方法使用Contextde 客户端的信息,你也可以通过Groups属性来管理组。
public abstract class Hub : IHub, IDisposable
{
public HubConnectionContext Clients { get; set; }
public HubCallerContext Context { get; set; }
public IGroupManager Groups { get; set; }
public virtual Task OnConnected();
public virtual Task OnDisconnected();
public virtual Task OnReconnected();
}
子类可以重写Onconnected,OnDisconnected方法,这个对事件触发一些动作是非常有用的,ChatHub 类是Hub类的子类,在接下来的部分我们来讨论它们的方法
public class ChatHub : Hub
{
public void Connect(string userName);
public void SendMessageToAll(string userName, string message);
public void SendPrivateMessage(string toUserId, string message);
public override System.Threading.Tasks.Task OnDisconnected();
.
.
.
}
客户端在他想进入聊天室的时候会调用Connect方法,调用SendMessageToAll方法,他可以向聊天室中保持连接的所有人
通过调用 SendPrivateMessage 方法,key与单独的一个客户端聊天
客户端的代理(Proxy)
在项目中添加index.html 页面,引用JQuery,SignalR、还有自动生成的hubs(?不知道这个指什么)
<script src="http://www.codeproject.com/Scripts/jquery-1.8.2.min.js%22%3E%3C/script>
<!--Reference the SignalR library. -->
<script src="http://www.codeproject.com/Scripts/jquery.signalR-1.0.0.js%22%3E%3C/script>
<!--Reference the autogenerated SignalR hub script. -->
<script src="http://www.codeproject.com/signalr/hubs%22%3E%3C/script>
在客户端,你需要创建一个hub 代理(hub proxy)并启动它,一可以通过$.connection.yourHubClass来创建它$.connection.hub.start() 来启动它
<script type="text/javascript">
$(function () {
// Declare a proxy to reference the hub.
var chatHub = $.connection.chatHub;
registerClientMethods(chatHub);
// Start Hub
$.connection.hub.start().done(function () {
registerEvents(chatHub)
});
});
</script>
在这里我们需要注意的是自动生成的Hub proxy的命名方法,我们在服务器端用Camel命名法给ChatHub类命名,但是在客户端我们得到的是用little camel 方式命名的,就像$.connection.chatHub
连接到聊天室

用户成功的链接之后发送他的名称就可以连接到聊天室,我们会给他发送一个列表,和一些存在我们程序中的最近的聊天记录在ChatHub类中的第一个方法就是Connect
public void Connect(string userName)
{
var id = Context.ConnectionId;
if (ConnectedUsers.Count(x => x.ConnectionId == id) == 0)
{
ConnectedUsers.Add(new UserDetail { ConnectionId = id, UserName = userName });
// send to caller
Clients.Caller.onConnected(id, userName, ConnectedUsers, CurrentMessage);
// send to all except caller client
Clients.AllExcept(id).onNewUserConnected(id, userName);
}
}
通过Context.ConnectionId属性拿到调用Connect方法的用户的连接Id,在已有的链接列表汇总核对是否已经存在该用户的链接,现在我们还要做两件事,首先,我们要将已连接的客户端的列表和最近的聊天记录发送给想要链接的客户端,然后我们要同通知其他的用户有新人加进来了,我们可以通过Clients.Caller属性来很方便的调用想要连接到聊天室的客户端的方法:
// send to caller
Clients.Caller.onConnected(id, userName, ConnectedUsers, CurrentMessage);
要通知其他的链接在聊天室的人,但是我们又不想调用新加入的人的方法,可以用Clients的AllExcept属性,它可以按照你的意愿来排除一些客户端
// send to all except caller client
Clients.AllExcept(id).onNewUserConnected(id, userName);
定义/发布 你在服务器端调用的客户端的方法,做法是声明Clients.Caller.onConnected(...) 和 Clients.AllExcept(id).onNewUserConnected(...)
???You can define your methods using chatHub.client.yourMethodName at client side.
// Calls when user successfully logged in
chatHub.client.onConnected = function (id, userName, allUsers, messages) {
.
.
.
}
// On New User Connected
chatHub.client.onNewUserConnected = function (id, name) {
AddUser(chatHub, id, name);
}
用chatHub.server.yourMethod声明, 在客户端,你可以调用服务器端的方法。
chatHub.server.connect(name);
在聊天室中发送消息:
在出聊天室中,用户输入的消息会广播给所有正在连接中的用户,在服务器端的ChatHub类中,写用Clients.All.messageReceived来写 SendMessageToAll方法,
messageReceived是一个客户端的方法。
public void SendMessageToAll(string userName, string message)
{
// store last 100 messages in cache
AddMessageinCache(userName, message);
// Broad cast message
Clients.All.messageReceived(userName, message);
}
在客户端引发messageReceived 方法,这个方式只是简单的用JQuery将消息加入到聊天区域
chatHub.client.messageReceived = function (userName, message) {
AddMessage(userName, message);
}
在写好了客户端和服务器端的方法以后,现在我们要在客户端来注册按钮的点击事件,在用户点击按钮的时候,就会调用服务器端的sendMessageToAll方法,它会广播消息给所有处于连接状态的客户端
$('#btnSendMsg').click(function () {
var msg = $("#txtMessage").val();
if (msg.length > 0) {
var userName = $('#hdUserName').val();
chatHub.server.sendMessageToAll(userName, msg);
$("#txtMessage").val('');
}
});
私聊模式
你也可以通过双击客户端的名称实现私聊,私聊的时候我们不用发信息给所有正在连接的客户端,在私聊的时候,只有两个客户端之间进行通话,所以在发送私人信息的时候,
发送人要调用sendPrivateMessage 方法
public void SendPrivateMessage(string toUserId, string message)
{
string fromUserId = Context.ConnectionId;
var toUser = ConnectedUsers.FirstOrDefault(x => x.ConnectionId == toUserId) ;
var fromUser = ConnectedUsers.FirstOrDefault(x => x.ConnectionId == fromUserId);
if (toUser != null && fromUser!=null)
{
// send to
Clients.Client(toUserId).sendPrivateMessage(fromUserId, fromUser.UserName, message);
// send to caller user
Clients.Caller.sendPrivateMessage(toUserId, fromUser.UserName, message);
}
}
断开连接:
在浏览器关闭的时候,SignalR API 调用 OnDisconnected 方法,在你的ChatHub类中重写这个方法,
在这个方法中将断开连接的客户端从缓存结合中删除并且向其他的用户发送一个提示。
public override System.Threading.Tasks.Task OnDisconnected()
{
var item = ConnectedUsers.FirstOrDefault(x => x.ConnectionId == Context.ConnectionId);
if (item != null)
{
ConnectedUsers.Remove(item);
var id = Context.ConnectionId;
Clients.All.onUserDisconnected(id, item.UserName);
}
return base.OnDisconnected();
}
SignalR 聊天室实例详解(服务器端推送版)的更多相关文章
- 当里个当,免费的HTML5连载来了《HTML5网页开发实例详解》连载(一)
读懂<HTML5网页开发实例详解>这本书 你还在用Flash嘛?帮主早不用了 乔布斯生前在公开信“Flash之我见”中预言:像HTML 5这样在移动时代中创立的新标准,将会在移动设备上获得 ...
- Cocos2d-x 3.X手游开发实例详解
Cocos2d-x 3.X手游开发实例详解(最新最简Cocos2d-x手机游戏开发学习方法,以热门游戏2048.卡牌为例,完整再现手游的开发过程,实例丰富,代码完备,Cocos2d-x作者之一林顺和泰 ...
- Linux下rz命令使用的实例详解
Linux中rz命令和sz命令都可用于文件传输,而rz命令主要用于文件的上传,下面将通过几个实例来给大家详细介绍下Linux下rz命令的用法,一起来学习下吧. rz命令可以批量上传文件,当然也可上传单 ...
- Vue 实例详解与生命周期
Vue 实例详解与生命周期 Vue 的实例是 Vue 框架的入口,其实也就是前端的 ViewModel,它包含了页面中的业务逻辑处理.数据模型等,当然它也有自己的一系列的生命周期的事件钩子,辅助我们进 ...
- 「微信小程序」PHP异步进程async-helper实例详解
PHP异步进程async-helper实例详解 PHP 的异步进程助手,借助于 AMQP 实现异步执行 PHP 的方法,将一些很耗时.追求高可用.需要重试机制的操作放到异步进程中去执行,将你的 HTT ...
- JS JSOP跨域请求实例详解
JSONP(JSON with Padding)是JSON的一种“使用模式”,可用于解决主流浏览器的跨域数据访问的问题.这篇文章主要介绍了JS JSOP跨域请求实例详解的相关资料,需要的朋友可以参考下 ...
- linux基础-磁盘阵列(RAID)实例详解
磁盘阵列(RAID)实例详解 raid技术分类 软raid技术 硬raid技术 Raid和lvm的区别 为什么选择用raid RAID详解 RAID-0 RAID-1 RAID-5 Raid-10 R ...
- JavaScript学习笔记-实例详解-类(二)
实例详解-类(二) //===给Object.prototype添加只读\不可枚举\不可配置的属性objectId(function(){ Object.defineProperty(Object ...
- JavaScript学习笔记-实例详解-类(一)
实例详解-类(一): //每个javascript函数(除了bind())都自动拥有一个prototype对象// 在未添加属性或重写prototype对象之前,它只包含唯一一个不可枚举属性const ...
随机推荐
- pip install 出现报asciii码错误的解决
原因是pip安装python包会加载我的用户目录,我的用户目录恰好是中文的,ascii不能编码.解决办法是: python目录 Python27\Lib\site-packages 建一个文件site ...
- [翻译]Django速查表
原文在此:https://code.djangoproject.com/wiki/DjangoCheatSheet Django速查表Django教程已经非常好了.这个速查表的作用是创建一个快速开始指 ...
- Mac编程(QT有许多专门的资料)
Mac OS X 上在应用运行时,在Dock上的图标右键会有额外的菜单部分.参考iTunes运行时右键的菜单.使用Qt在Mac下的一个set_menu(QMenu *)函数实现,文档里有写 http: ...
- QtInternal 之 高效使用QString(使用QLatin1String,QStringRef,QStringBuilder,QStringMatcher等相关类)
注意:本文翻译自 http://developer.qt.nokia.com 中的 UsingQStringEffectively ,中文译文见 简体中文版 ,如果你对翻译wiki感兴趣 ...
- HttpGet()和HttpPost()2
Get一般用于从服务器取数据,而且不改变原来的内容: Post一般用于向服务器传递数据,这需要改变服务器的内容. 从安全性上考虑,Get的安全性要稍微差点,因为它会把信息直接在地址栏显示出来.(但是A ...
- extjs 优化小建议
1 原文信息 原文标题: Sencha Con 2013: Ext JS Performance tips 原文地址: [http://edspencer.net/2013/07/19/sencha- ...
- Button的设置及各种属性
(1)UIButton类继承自UIControl,而UIControl继承自UIView,因为UIView就是个矩形区域,所以UIButton实例化的对象其实都是一个矩形,虽然有各种圆角.增加联系人. ...
- Lucene 实例教程(二)
原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本人声明.否则将追究法律责任. 作者: 永恒の_☆ 地址: http://blog.csdn.net/chenghui031 ...
- 【UVA】658 - It's not a Bug, it's a Feature!(隐式图 + 位运算)
这题直接隐式图 + 位运算暴力搜出来的,2.5s险过,不是正法,做完这题做的最大收获就是学会了一些位运算的处理方式. 1.将s中二进制第k位变成0的处理方式: s = s & (~(1 < ...
- XMPP个人信息展示
在现阶段的通信服务中.各种标准都有,因此会出现无法实现相互连通,而XMPP(Extensible Message and presence Protocol)协议的出现.实现了整个及时通信服务协议的互 ...