接上一篇,文章末尾抛出了2个问题:

  1. 能不能让客户端声明一个强类型的方法列表呢?这样首先不容易写错。
  2. 同样的,能不能让服务端声明一个强类型的方法列表给客户端调用呢?

如果要让客户端的方法以强类型出现在服务端,同样的,服务端的方法也以强类型出现在客户端,那就必须声明类似契约一样的载体。比如:

public interface IChatClient
{
void broadcast(string name, string message);
}
public interface IChatHub
{
void Send(string name, string message);
}

分别建立ChatClient接口和ChatHub的接口。

public class ChatHub : Hub<IChatClient>
{
...
}

这是最终的目标,一个泛型Hub。

好,现在需要进行一些分析,怎样才能让Hub支持泛型。

首先,看一下Hub是如何操作客户端方法的:

Clients.AllExcept(Context.ConnectionId).broadcast(name, message);

Hub通过Clients来操作所有客户端的行为。那么这个Clients又是什么类型的呢?

// 摘要:
// Gets a dynamic object that represents all clients connected to this hub (not
// hub instance).
IHubCallerConnectionContext Clients { get; set; }

通过IHub接口看到,Clients的类型是IHubCallerConnectionContext,点进去看:

// 摘要:
// Encapsulates all information about an individual SignalR connection for an
// Microsoft.AspNet.SignalR.Hubs.IHub.
public interface IHubCallerConnectionContext : IHubConnectionContext
{
[Dynamic]
dynamic Caller { get; }
[Dynamic]
dynamic Others { get; } dynamic OthersInGroup(string groupName);
dynamic OthersInGroups(IList<string> groupNames);
}

IHubCallerConnectionContext又继承IHubConnectionContext,再点进去看:

// 摘要:
// Encapsulates all information about a SignalR connection for an Microsoft.AspNet.SignalR.Hubs.IHub.
public interface IHubConnectionContext
{
[Dynamic]
dynamic All { get; } dynamic AllExcept(params string[] excludeConnectionIds);
dynamic Client(string connectionId);
dynamic Clients(IList<string> connectionIds);
dynamic Group(string groupName, params string[] excludeConnectionIds);
dynamic Groups(IList<string> groupNames, params string[] excludeConnectionIds);
dynamic User(string userId);
}

一目了然,所有Clients的操作方法都在这儿了,全是动态类型的,这也是为什么在Hub中写到Clients.All.xxx的时候已经是动态的了,那么运行时,这些操作都是什么类型的呢?试一下:

运行时,Clients的操作返回的是ClientProxy类型,从代码中扒出来:

public class ClientProxy : DynamicObject, IClientProxy
{
public ClientProxy(IConnection connection, IHubPipelineInvoker invoker, string hubName, IList<string> exclude); public Task Invoke(string method, params object[] args);
public override bool TryInvokeMember(InvokeMemberBinder binder, object[] args, out object result);
}
// 摘要:
// A server side proxy for the client side hub.
public interface IClientProxy
{
// 摘要:
// Invokes a method on the connection(s) represented by the Microsoft.AspNet.SignalR.Hubs.IClientProxy
// instance.
//
// 参数:
// method:
// name of the method to invoke
//
// args:
// argumetns to pass to the client
//
// 返回结果:
// A task that represents when the data has been sent to the client.
Task Invoke(string method, params object[] args);
}
}

可以看到,运行时如果以IClientProxy注入,就一个Invoke方法。

好,挖到这儿,可以有一些思路了。

  1. Clients所有的操作最终都是通过IClientProxy的Invoke来执行的。
  2. 如果让IChatClient通过某种方式和IClientProxy建立起非运行时的联系,就能实现强类型了。
  3. 这样的话,就需要有一个Hub<T>的类,然后把Clients里所有的操作在Hub<T>中重新实现一次。
  4. 然后T又是客户端的行为接口,因此,需要对Hub<T>进行静态扩展,让IClientProxy的Invoke方法能够被T的所有方法自动调用。

核心攻克点找到了,解决了4,就能一路解决1。怎样才能让IClientProxy的Invoke自动的被T的所有方法调用呢?AOP可以!可以用Castle对T进行动态织入。到这儿可以动手了,先建立一个Hub扩展类:

public static class HubExtensions
{
static readonly ProxyGenerator generator = new ProxyGenerator(); public static T GetClientBehavior<T>(this IClientProxy clientProxy) where T : class
{
return (T)generator.CreateInterfaceProxyWithoutTarget<T>(new ClientBehaviorInterceptor(clientProxy));
}
}

让所有的IClientProxy执行GetClientBehavior方法,然后内部进行拦截器装载,并将IClientProxy塞进拦截器。

public class ClientBehaviorInterceptor:IInterceptor
{
public ClientBehaviorInterceptor(IClientProxy clientProxy)
{
this.clientProxy = clientProxy;
} IClientProxy clientProxy; public void Intercept(IInvocation invocation)
{
clientProxy.Invoke(invocation.Method.Name, invocation.Arguments);
}
}

拦截器中,每当T执行方法的时候,clientProxy就执行Invoke方法,把T的方法名和T的参数传入,这就达到了原先动态调用客户端方法传入参数并执行的效果。

然后就是写一个Hub<T>了。

public abstract class Hub<T> : Hub where T : class
{
protected T All { get { return (Clients.All as IClientProxy).GetClientBehavior<T>(); } } protected T Any(params string[] connectionIds)
{
return (Clients.Clients(connectionIds) as IClientProxy).GetClientBehavior<T>();
} protected T Except(params string[] connectionIds)
{
return (Clients.AllExcept(connectionIds) as IClientProxy).GetClientBehavior<T>();
} protected T Client(string connectionId)
{
return (Clients.Client(connectionId) as IClientProxy).GetClientBehavior<T>();
} protected T Caller { get { return (Clients.Caller as IClientProxy).GetClientBehavior<T>(); } }
}

把Clients中所有的操作都在这儿写一遍,例子中就写了5个。通过刚才的扩展方法,返回的T已经是经过AOP的了。最后,把最初的ChatHub改一下:

让ChatHub继承Hub<T>,T为IChatClient,如图示,已经可以通过Except方法用强类型调用客户端方法了。执行一下看看:

到此,服务端改造结束。服务端已经可以接受强类型的客户端行为。

下一篇将对客户端部分进行强类型改造。

最后附上一个基于SignalR的聊天室玩具,绿色无毒:http://www.royarea.cn/chatroom

转载请注明出处:http://www.cnblogs.com/royding/p/3750412.html

 
 
 
标签: SignalRAOP

最新文章

  1. 使用MonkeyTest对Android客户端进展压力测试
  2. Spark HA实战
  3. Android中的内容提供器
  4. 将jar包直接Buldpath所引起的问题
  5. SQL MD5加密
  6. Dynamic CRM 2013学习笔记(三十四)自定义审批流5 - 自动邮件通知
  7. [转]Sublime Text3注册码(可用)
  8. 自己实现FormsAuthentication.SetAuthCookie方法,怎样在ASP.NET服务端代码中删除客户端Cookie
  9. sass初步认识3
  10. 黄聪:wordpress/wp-admin目录文件
  11. [itint5]单词变换
  12. Web调用安卓,苹果手机摄像头,本地图片和文件
  13. Slice到C++映射
  14. ORACLE - 管理重做日志文件
  15. MySQL一对一:一对多:多对多: 实例!!!!
  16. P1742 最小圆覆盖(计算几何)
  17. windows下vmware配置nat网络
  18. [CF528D]Fuzzy Seach
  19. 一: vue的基本使用
  20. 集群同步hive的脚本

热门文章

  1. python有些错误换行问题解决
  2. JAVA连接ACCESS、MYSQL、SQLSEVER、ORACLE数据库
  3. ExtJs在disabled和readOnly美学分析
  4. NYNU_省赛选拔题(5)
  5. C# 你不能调用的问题剪贴板线程
  6. javascript/jquery读取和修改HTTP headers
  7. 使用 CXF 做 webservice 简单例子[转]
  8. Apache conf文件配置个人总结
  9. jQuery性能优化38建议---最引人注目的用户体验!
  10. Jquery()核心函数的7个重载方法