原文:一步一步学习SignalR进行实时通信_5_Hub

一步一步学习SignalR进行实时通信\_5_Hub

SignalR


前言

上一讲,我们简单的介绍了下Hub的配置以及实现方法,这一将我希望把更多的细节梳理清楚,不至于让大家在细节上面摸不着头脑,理解深了,那么做项目自然就会相对轻松一些。

Hub命名规则

Hub与PersistentConnection很大的区别就是我们可以自己定义我们自己的方法,取我们想取得名字。

Hub的前台调用后台方法的命名规则遵循骆驼命名法,如果不遵循该约定,那么在程序中很有可能报错,在.net中约定大于配置已经是被人们普遍接受的了,它的好处是免去了大量的配置文件,而用一个公用的约定来完成,每个人只要遵循这个约定就不会出篓子。

比如在后台有这么一个方法

  1. public void SendMessageByUserName(string userName)
  2. {
  3. ...
  4. }

那么在前台通过hub.server.sendMessageByUserName("name")调用

如果后台方法改为

  1. public void sendMessageByUserName(string userName)
  2. {
  3. ...
  4. }

在前台则用hub.server.sendmessageByUserName("name")调用以此类推

在后台调用前台方法则无此约定,不区分大小写。因此假设后台你通过Clients.Caller.testHtmlClient();调用前台的方法前台你可以写为hub.client.testHtmlClient(),也可以写为hub.client.testhtmlclient()甚至可以写为hub.client.teStHtMlClIENT()

Hub封装好的常用方法

假如我希望给指定姓名的人发送信息,Hub可以通过唯一的ConnectionId号发送给指定客户端,但是我们一般都是根据数据库中保存用户名或者数据库每条记录的Id进行数据操作。我们希望根据用户名来发送,那么我假设有一个字典保存了ConnectionId和UserName的对应关系

  1. public class MySecondHub : Hub
  2. {
  3. private IDictionary<string, string> _userNames;
  4. public MySecondHub()
  5. {
  6. //TODO:初始化UserNames
  7. }
  8. public void SendMessageByUserName(string userName)
  9. {
  10. //取到所有名字为那么的用户
  11. IList<string> users = _userNames.Where(u => u.Value == userName).Select(u => u.Key).ToList();
  12. Clients.Clients(users).sendMessage("Hi!");
  13. }
  14. }

那么在html页面我们可以通过 hub.server.sendMessageByUserName()调用

  1. <script type="text/javascript">
  2. $(function () {
  3. //创建一个hub服务
  4. var hub = $.connection.myFirstHub;
  5. $.connection.hub.start()
  6. .done(function () {
  7. alert("连接成功!");
  8. })
  9. $("#sayHello").click(function () {
  10. hub.server.sendMessageByUserName("Jake");
  11. });
  12. hub.client.sendMessageByUserName = function(){
  13. ...
  14. };
  15. });
  16. </script>

我们也可以修改前台调用方法的名字

  1. ...
  2. //在前台可以通过sendMessageToGroup()调用
  3. [HubMethodName("SendMessageToGroup")]
  4. public void SendMessageByGroupName(string groupName)
  5. {
  6. Clients.Group(groupName);
  7. }

这样在客户端我们就可以通过sendMessageToGroup()调用SendMessageByGroupName()方法了。

同时,我们传递的参数但可以是字符串也可以是复杂的类型,如:

  1. public class Person
  2. {
  3. public string Name { get; set; }
  4. public int Age { get; set; }
  5. }
  1. public void SayHi(string name,string message)
  2. {
  3. var person = new Person(){Name = name,Message = message};
  4. Clients.Others.sayHi(person);
  5. }

前台代码,点击id为sayHello的按钮触发

  1. ...
  2. $("#sayHello").click(function () {
  3. hub.server.sayHi("jake","hah~");
  4. });
  5. hub.client.sayHi = function(person) {
  6. $('#chat').append('<li><strong>' + person.Message + '</strong>:' + person.Name + '</li>');
  7. };
  8. ...

之前我们讲到Client.Others.doSomething();可以调用给报包括自己的所有其他客户端doSomething()方法,Hub还有许多其他的方法供我们调用

Hub常用方法解释

  • Clients.Caller:可以与调用者进行通信
  • Clients.Others:可以与除了自己以外所有连接上此Hub的客户端的通信
  • Clients.All:可以与所有连接上此Hub的客户端通信
  • Clients.OthersInGroup:可以与指定组以外的其他连接到Hub的客户端通信
  • Clients.Client:可以与给指定ConnectionId的客户端进行通信
  • Clients.AllExcept:可以与所有连接上此Hub但是除去指定ConnectionId以外的客户端通信
  • Clients.Group:可以与在指定组的客户端通信
  • Clients.User:可以与指定的userId进行通信

    当然他们还有一些重载方法,这里就不一一介绍了,自己去试验下就能明白了。

    PS:这里补充一个小知识,在MVC中已经实现了获取默认的UserId方法

  1. public class PrincipalUserIdProvider : IUserIdProvider
  2. {
  3. public string GetUserId(IRequest request)
  4. {
  5. if (request == null)
  6. throw new ArgumentNullException("request");
  7. if (request.User != null && request.User.Identity != null)
  8. return request.User.Identity.Name;
  9. else return (string) null;
  10. }
  11. }

很多时候我们可能需要自己实现自己的方法,比如userId为登陆的userName,那么我们可以实现IUserIdProvider接口

  1. public class CookiesUserIdProvider : IUserIdProvider
  2. {
  3. public string GetUserId(IRequest request)
  4. {
  5. if (request == null)
  6. throw new ArgumentNullException("request");
  7. Cookie cookie;
  8. if (request.Cookies.TryGetValue("UserName", out cookie))
  9. {
  10. return cookie.Value;
  11. }
  12. else
  13. {
  14. return null;
  15. }
  16. }
  17. }

当然我们还要告诉我们的项目使用我们自定义的提供器,而不是默认的,我们只需要在程序一开始,也就是Global中注册代码保证程序启动时调用即可。

  1. public class MvcApplication : System.Web.HttpApplication
  2. {
  3. protected void Application_Start()
  4. {
  5. ...
  6. //注册我们自己的Id提供器
  7. GlobalHost.DependencyResolver.Register(typeof (IUserIdProvider), () => new CookiesUserIdProvider());
  8. }
  9. }
  10. }

保持状态

在hub中可以通过hub.state保存用户的状态



后台通过Client.Caller 获取前台传递来的参数运行项目断点,可以看到接收到的数据



参数名可以任意写,它是dynamic类型

前后台交互

难道保存状态只有这个用处吗?当然不止,这个状态还可以在某些情况下起到前后台交互的作用。

后台我们让Age++



然后前台在回调方法中我们在控制台中打印age,如果打印的是24,那么证明前后台数据能进行很好的交互而不仅仅是保存一个状态那么简单。



点击按钮后 通过firebug在控制台中可以看到打印出的是24

结束语

今天的文章知识点可能较零散,因为并没有一个实际例子来连接所有的知识,明天开始放假了,如果有时间的话希望能讲一个小例子来巩固一下。

源码下载

由于今天学习的都是些小细节,就不提供源码了。

本文发布至作业部落

参考文献

SignalR Programming in Microsoft ASP.NET pdf 下载

一步一步学习SignalR进行实时通信_5_Hub的更多相关文章

  1. 一步一步学习SignalR进行实时通信_1_简单介绍

    一步一步学习SignalR进行实时通信\_1_简单介绍 SignalR 一步一步学习SignalR进行实时通信_1_简单介绍 前言 SignalR介绍 支持的平台 相关说明 OWIN 结束语 参考文献 ...

  2. 一步一步学习SignalR进行实时通信_8_案例2

    原文:一步一步学习SignalR进行实时通信_8_案例2 一步一步学习SignalR进行实时通信\_8_案例2 SignalR 一步一步学习SignalR进行实时通信_8_案例2 前言 配置Hub 建 ...

  3. 一步一步学习SignalR进行实时通信_9_托管在非Web应用程序

    原文:一步一步学习SignalR进行实时通信_9_托管在非Web应用程序 一步一步学习SignalR进行实时通信\_9_托管在非Web应用程序 一步一步学习SignalR进行实时通信_9_托管在非We ...

  4. 一步一步学习SignalR进行实时通信_7_非代理

    原文:一步一步学习SignalR进行实时通信_7_非代理 一步一步学习SignalR进行实时通信\_7_非代理 SignalR 一步一步学习SignalR进行实时通信_7_非代理 前言 代理与非代理 ...

  5. 一步一步学习SignalR进行实时通信_6_案例

    原文:一步一步学习SignalR进行实时通信_6_案例 一步一步学习SignalR进行实时通信\_6_案例1 一步一步学习SignalR进行实时通信_6_案例1 前言 类的定义 各块功能 后台 上线 ...

  6. 一步一步学习SignalR进行实时通信_4_Hub

    原文:一步一步学习SignalR进行实时通信_4_Hub 一步一步学习SignalR进行实时通信\_4_Hub SignalR 一步一步学习SignalR进行实时通信_4_Hub 前言 创建Hub 配 ...

  7. 一步一步学习SignalR进行实时通信_3_通过CORS解决跨域

    原文:一步一步学习SignalR进行实时通信_3_通过CORS解决跨域 一步一步学习SignalR进行实时通信\_3_通过CORS解决跨域 SignalR 一步一步学习SignalR进行实时通信_3_ ...

  8. 一步一步学习SignalR进行实时通信_2_Persistent Connections

    原文:一步一步学习SignalR进行实时通信_2_Persistent Connections 一步一步学习SignalR进行实时通信\_2_Persistent Connections Signal ...

  9. 12.Linux软件安装 (一步一步学习大数据系列之 Linux)

    1.如何上传安装包到服务器 有三种方式: 1.1使用图形化工具,如: filezilla 如何使用FileZilla上传和下载文件 1.2使用 sftp 工具: 在 windows下使用CRT 软件 ...

随机推荐

  1. 正三角形的外接圆面积,nyoj-274

    正三角形的外接圆面积 时间限制:1000 ms  |  内存限制:65535 KB 难度:0 描述给你正三角形的边长,pi=3.1415926 ,求正三角形的外接圆面积.   输入 只有一组测试数据 ...

  2. codeforces 519A. A and B and Chess,

    A. A and B and Chess time limit per test 1 second memory limit per test 256 megabytes input standard ...

  3. SQL Server 备份的 8 种方法。

    方法 1. 完整备份 方法 2. 差异备份 方法 3. 部分备份(备份数据库的read_write部分) 方法 4. 文件备份 方法 5. 文件组备份 方法 6. 只复制备份 方法 7. 日志备份 - ...

  4. Dispatcher.BeginInvoke()方法使用不当导致UI界面卡死的原因分析

    原文:Dispatcher.BeginInvoke()方法使用不当导致UI界面卡死的原因分析 前段时间,公司同事开发了一个小工具,在工具执行过程中,UI界面一直处于卡死状态. 通过阅读代码发现,主要是 ...

  5. 如何让FPGA中的SPI与其他模块互动起来

    在上一篇文章<FPGA的SPI从机模块实现>中,已经实现了SPI的从机模块,如何通过SPI总线与FPGA内部其他模块进行通信,是本文的主要讨论内容. 一. 新建FPGA内部DAC控制模块 ...

  6. hibernate 非xml实体类配置方法!

    hibernate 非xml实体类配置方法! 这个是hibernate.cfg.xml配置文件 <?xml version='1.0' encoding='UTF-8'?> <!DO ...

  7. [转]Geoserver实现WFS操作

    From:http://liushaobo2005.blog.163.com/blog/static/253056702011541462372/ wfs是OGC的标准规范,主要用于提供对矢量地理数据 ...

  8. [分享]源代码&开发手记:SAE应用“车百科” (Python + SAE + Bottle + Bootstrap) - Bottle - Python4cn(news, jobs)

    [分享]源代码&开发手记:SAE应用"车百科" (Python + SAE + Bottle + Bootstrap) - Bottle - Python4cn(news, ...

  9. Windows.Server.2003.R2 简体中文企业版 x86 x64(转)

    两张盘,第二张是 R2安装盘. Windows.Server.2003.R2.With.Sp2 中文企业版[MSDN官方版本][32bit] Windows 2003.R2.With.Sp2 简体中文 ...

  10. Echart的angularjs封装

    ehcart是百度做的数据图表,基于原生js.接口和配置都写的很好很易读,还可以用于商用. 下面正题 用原生js的话,引入echarts.js 无论是图表的样式设置,图表渲染,数据填充都是基于echa ...