博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
一步一步学习SignalR进行实时通信_5_Hub
阅读量:6183 次
发布时间:2019-06-21

本文共 4079 字,大约阅读时间需要 13 分钟。

目录

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

标签(空格分隔): SignalR


前言

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

Hub命名规则

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

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

public void SendMessageByUserName(string userName){    ...}

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

如果后台方法改为

public void sendMessageByUserName(string userName){    ...}

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

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

Hub封装好的常用方法

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

public class MySecondHub : Hub{    private IDictionary
_userNames; public MySecondHub() { //TODO:初始化UserNames } public void SendMessageByUserName(string userName) { //取到所有名字为那么的用户 IList
users = _userNames.Where(u => u.Value == userName).Select(u => u.Key).ToList(); Clients.Clients(users).sendMessage("Hi!"); } }

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

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

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

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

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

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

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

...$("#sayHello").click(function () {    hub.server.sayHi("jake","hah~");});hub.client.sayHi = function(person) {    $('#chat').append('
  • ' + person.Message + ':' + person.Name + '
  • ');};...

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

    40473958.jpg

    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方法

    public class PrincipalUserIdProvider : IUserIdProvider{    public string GetUserId(IRequest request)     {        if (request == null)             throw new ArgumentNullException("request");        if (request.User != null && request.User.Identity != null)             return request.User.Identity.Name;         else return (string) null;     }}

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

    public class CookiesUserIdProvider : IUserIdProvider {    public string GetUserId(IRequest request)     {        if (request == null)             throw new ArgumentNullException("request");        Cookie cookie;         if (request.Cookies.TryGetValue("UserName", out cookie))         {            return cookie.Value;         }        else        {            return null;         }        }}

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

    public class MvcApplication : System.Web.HttpApplication    {        protected void Application_Start()        {            ...            //注册我们自己的Id提供器            GlobalHost.DependencyResolver.Register(typeof (IUserIdProvider), () => new CookiesUserIdProvider());        }    }}

    保持状态

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

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

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

    前后台交互

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

    后台我们让Age++
    50259171.jpg
    然后前台在回调方法中我们在控制台中打印age,如果打印的是24,那么证明前后台数据能进行很好的交互而不仅仅是保存一个状态那么简单。
    39635239.jpg
    点击按钮后 通过firebug在控制台中可以看到打印出的是24
    95170579.jpg

    结束语

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

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

    本文发布至

    参考文献

    转载于:https://www.cnblogs.com/Jack-Blog/p/4779779.html

    你可能感兴趣的文章
    分布式锁
    查看>>
    <![CDATA[ ]]>
    查看>>
    HP Gen8 + Epson L211 + Ubuntu 搭建内部服务器
    查看>>
    UITextView字数限制自适应高度限制输入
    查看>>
    php 跨域、跨子域,跨服务器读取session的方法介绍
    查看>>
    适配器模式
    查看>>
    利用btrace在线监控java程序状态
    查看>>
    多线程求大数组的最大值
    查看>>
    实战做项目如何选择开源许可协议(二)- 开放代码
    查看>>
    4.3 Hibernate关系映射
    查看>>
    IO复用\阻塞IO\非阻塞IO\同步IO\异步IO
    查看>>
    Canopy聚类算法
    查看>>
    高并发的大型网站架构设计
    查看>>
    Implement Magic Dictionary
    查看>>
    Java提高篇(三一)—–Stack
    查看>>
    关于java web项目使用log4j
    查看>>
    AndroidPN系列之三--服务端Bug汇总
    查看>>
    【转载】关于时间、时区、系统时间和硬件时间
    查看>>
    Java Class.forName、Class.class、getClass示例区分
    查看>>
    为什么会"well-known file is not secure" ?
    查看>>