初探Remoting双向通信(四)
初探Remoting双向通信(四)
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/kkkkkxiaofei/article/details/9175429 </div>
<link rel="stylesheet" href="https://csdnimg.cn/release/phoenix/template/css/ck_htmledit_views-cd6c485e8b.css">
<link rel="stylesheet" href="https://csdnimg.cn/release/phoenix/template/css/ck_htmledit_views-cd6c485e8b.css">
<div class="htmledit_views" id="content_views">
之前已经从基本原理上实现了Remoting的双向通信。准备将其移植到我的项目中,不过为了成功移植,我还是需要再把以前的版本稍作修改才能放心的去做。项目中当一台机子中有工作人员进行了预警信息标记时(在地图上会有一个标志),其他机子需要同步更新。这就是说任何一个客户端进行标记时需要通知服务器,然后让服务器去广播标记。这一个小动作就需要双向通信一次。按照项目需求,我再次修改版本,这次只需要改一个函数:
修改服务器订阅事件函数如下:
-
void marshal_obj_SubscribeAtServer(string msg)
-
{
-
//跨线程调用
-
textBox2.Invoke(new Action<string>(str => { textBox2.AppendText(str); }), msg);
-
//订阅函数不在主线程,需另起线程触发广播方法
-
Thread th = new Thread(new ThreadStart(() => {
-
marshal_obj.TriggerAtServer("服务器--" + this.ServerIP() + System.Environment.NewLine + "我是服务器,我发现一个客户端有标记行为,我要让你们全部都标记!" + System.Environment.NewLine);
-
}));
-
th.Start();
-
}
然后开启一个服务器3个客户端,让其中一个客户端发起标记事件,效果如下:
有了这个,我更加肯定的去修改自己的项目了。经过一整天的大改,我把项目调试通了,开了1个服务器,2个客户端,分别设置了断点,虽然其中事件订阅处有很多逻辑处理,并不像上面DEMO那样只发送一段文字,但是每一个断点都可以运行到,那就说明整体上是通了。删除断点后,很高兴的看一下新的效果时,却让我大失所望:
当一个客户端进行标记时,第二个客户端确实也收到了信息,但是第一个客户端界面已经卡死,很久才能恢复,甚至当机。这就让我想起了上面这个Demo运行的效果时的现象了。上面的三个客户端收到信息并不是同时的,而是依次的。况且,所谓的广播发送,其实利用的就是多播委托,并不像Socket那样是3个tcpclient来进行通信的。服务器端利用Marshal得到的对象和3个客户端激活得到的对象是同一个引用(SingleTon方式),所以客户端订阅的事件都是在服务器同一个对象上的委托列表里。当直接调用委托的时候,调用的是委托的Invoke方法,该方法会在执行的时候阻塞当前线程,直到委托列表执行完才会将控制权转回来。这样一来,委托的超时必然会引起界面的假死(当有更多逻辑处理在事件订阅函数里时更糟糕)。呢有什么办法呢,查找了资料,看了"张子阳"的委托与事件觉得很不错,仔细拜读了好几遍。原来委托执行的列表是可以获取的,而且可以认为的去遍历,这么一来就可以解决许多问题:比如有一个客户端发生了异常,为了不影响其他的客户端,可以遍历的时候try/catch掉;再比如遍历的时候为了不阻塞客户端线程,可以每次另起一个线程(这里就用到了BeginInvoke,它就相当于另起线程执行委托,只不过它用的是线程池)。好了,再次修改以下服务器的订阅函数,如下:
-
void marshal_obj_SubscribeAtServer(string msg)
-
{
-
//跨线程调用
-
textBox2.Invoke(new Action<string>(str => { textBox2.AppendText(str); }), msg);
-
//遍历委托列表
-
Delegate[] del_list = marshal_obj.GetServerEventList();
-
foreach (Delegate del in del_list)
-
{
-
MyDelegate d = (MyDelegate)del;
-
d.BeginInvoke(
-
"服务器--" + this.ServerIP() + System.Environment.NewLine + "我是服务器,我发现一个客户端有标记行为,我要让你们全部都标记!" + System.Environment.NewLine,
-
null,
-
null);
-
}
-
}
其中GetServerEventList()这个方法是在远程对象里新加的:
-
//获取委托列表
-
public Delegate[] GetServerEventList()
-
{
-
return this.SubscribeAtClient.GetInvocationList();
-
}
这次,我开了很多客户端,接收到信息几乎是同步的(至少肉眼分辨不出来),并且将项目也进行了改进,也有了重大突破。感谢网上许多乐于分享的大牛们,Remoting的学习暂时告一段落,还有其他事情要做,前方的路已然坎坷,而我却乐此不疲。。。。。
(注:上面的Demo在此处下载http://download.csdn.net/detail/kkkkkxiaofei/5650187)
初探Remoting双向通信(四)的更多相关文章
- 初探Remoting双向通信(三)
原 初探Remoting双向通信(三) 2013年06月25日 17:51:08 喜欢特别冷的冬天下着雪 阅读数 4741 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blo ...
- 初探Remoting双向通信(二)
原 初探Remoting双向通信(二) 2013年06月25日 11:46:24 喜欢特别冷的冬天下着雪 阅读数 2977 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blo ...
- 初探remoting双向通信(一)
原 初探remoting双向通信(一) 2013年06月24日 15:47:07 喜欢特别冷的冬天下着雪 阅读数 4389 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blo ...
- C# 实现Remoting双向通信
本篇文章主要介绍了C# 实现Remoting双向通信,.Net Remoting 是由客户端通过Remoting,访问通道以获得服务端对象,再通过代理解析为客户端对象来实现通信的 闲来无事想玩玩双向通 ...
- C# Remoting双向通信
闲来无事想玩玩双向通信,实现类似QQ的互发消息的功能.于是乎开始学习.Net Remoting. .Net Remoting 是由客户端通过Remoting,访问通道以获得服务端对象,再通过代理解析为 ...
- 【Nacos】本地集群部署
关于Nacos已经展开了四篇入门文章: 初探Nacos(一)-- 单机模式启动 初探Nacos(二)-- SpringCloud使用Nacos的服务注册与发现 初探Nacos(三)-- SpringB ...
- C#进阶系列——DDD领域驱动设计初探(四):WCF搭建
前言:前面三篇分享了下DDD里面的两个主要特性:聚合和仓储.领域层的搭建基本完成,当然还涉及到领域事件和领域服务的部分,后面再项目搭建的过程中慢慢引入,博主的思路是先将整个架构走通,然后一步一步来添加 ...
- 【原创】NIO框架入门(四):Android与MINA2、Netty4的跨平台UDP双向通信实战
概述 本文演示的是一个Android客户端程序,通过UDP协议与两个典型的NIO框架服务端,实现跨平台双向通信的完整Demo. 当前由于NIO框架的流行,使得开发大并发.高性能的互联网服务端成为可能. ...
- 初探JavaScript(四)——作用域链和声明提前
前言:最近恰逢毕业季,千千万万的学生党开始步入社会,告别象牙塔似的学校生活.往往在人生的各个拐点的时候,情感丰富,感触颇深,各种对过去的美好的总结,对未来的展望.与此同时,也让诸多的老“园”工看完这些 ...
随机推荐
- python基础----斐波那契数列
python实现斐波那契数列的三种方法 """ 斐波那契数列 0,1,1,2,3,5,8,13,21,... """ # 方法一:while ...
- 图片查看器(类似于QQ,另外又加了JARA的下方的图片缩略导航图)
源码地址:https://gitee.com/yolanda624/coffer/tree/master/src/components/a-photo-view
- 【串线篇】概述SpringMvc和spring整合
SpringMVC和Spring整合的目的:分工明确: SpringMVC的配置文件就来配置和网站转发逻辑以及网站功能有关的(视图解析器,文件上传解析器,支持ajax,xxx):springmvc.x ...
- Python3.5-20190518-廖老师-自我笔记-模块
在Python中,一个.py文件就称之为一个模块(Module) 可以作为module的文件类型有".py".".pyo".".pyc".& ...
- ECUST_Algorithm_2019_4
简要题意及解析 1001 第三次作业原题,略. 1002 把一个数转换为二进制. 不断除以\(2\)取余就好了.写递归代码会非常短. 时间复杂度\(O(Tlogn)\) \(T\)是数据组数,\(n\ ...
- python核心编程socket备忘
服务器端: # Echo server program from socket import * from time import ctime HOST = '' # Symbolic name me ...
- eclipse设置tomcat部署目录地址
参考: https://blog.csdn.net/lvyuan1234/article/details/53418818 右键,open 操作前提是所有项目移除,并且右键clean掉相关数据! 修改 ...
- boost库:事件处理
boost库的signal所实现的模式被命名为信号至插槽,当对应的信号被发出时,相关联的插槽即被执行. #include <boost/signal.hpp> #include <i ...
- delphi 监控文件系统
elphi 监控文件系统 你是否想为你的Windows加上一双眼睛,察看使用者在机器上所做的各种操作(例如建立.删除文件:改变文件或目录名字)呢? 这里介绍一种利用Windows未公开函数实现这个功能 ...
- 3D打印切片软件Cura及CuraEngine原理分析
引言 年初开始进入3D打印行业,受命以Cura为基础,研发一款自主的3D打印切片软件. 自主研发要取其长处,补其不足,首先自然是要搞清楚Cura到底做了什么,读Cura的代码是必需的.我一向都觉得比起 ...