初探Remoting双向通信(二)
初探Remoting双向通信(二)
版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/kkkkkxiaofei/article/details/9167829 </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">
二、利用事件实现客户端向服务器通信
接着上一篇,按照我的思路,远程对象中定义的事件在客户端触发,而在服务器端订阅,应该可以成功。现在放弃之前的示例代码,自己重新写一个版本,代码如下:
远程对象:
-
using System;
-
using System.Collections.Generic;
-
using System.Linq;
-
using System.Text;
-
-
namespace Remoting
-
{
-
-
public class RemotingObject:MarshalByRefObject
-
{
-
public delegate void MyDelegate(string msg);
-
public static event MyDelegate myEvent;
-
//触发事件
-
public void TriggerEvent(string msg)
-
{
-
if (myEvent != null)
-
myEvent(msg);
-
}
-
//无限生命周期
-
public override object InitializeLifetimeService()
-
{
-
return null;
-
}
-
}
-
}
服务器:
-
using System;
-
using System.Collections.Generic;
-
using System.ComponentModel;
-
using System.Data;
-
using System.Drawing;
-
using System.Linq;
-
using System.Text;
-
using System.Windows.Forms;
-
using System.Runtime.Remoting;
-
using System.Runtime.Remoting.Channels;
-
using System.Runtime.Remoting.Channels.Tcp;
-
using Remoting;
-
namespace Server
-
{
-
public partial class ServerForm : Form
-
{
-
public ServerForm()
-
{
-
InitializeComponent();
-
StartServer();
-
}
-
//开启服务器
-
public void StartServer()
-
{
-
TcpChannel tcpchannel = new TcpChannel(8080);
-
ChannelServices.RegisterChannel(tcpchannel,false);
-
-
RemotingConfiguration.RegisterWellKnownServiceType(typeof(RemotingObject),"url",WellKnownObjectMode.Singleton);
-
-
RemotingObject.myEvent += new RemotingObject.MyDelegate(RemotingObject_myEvent);
-
}
-
-
void RemotingObject_myEvent(string msg)
-
{
-
//跨线程调用
-
textBox2.Invoke(new Action<string>(str => { textBox2.Text = str; }), msg);
-
}
-
}
-
}
客户端:
-
using System;
-
using System.Collections.Generic;
-
using System.ComponentModel;
-
using System.Data;
-
using System.Drawing;
-
using System.Linq;
-
using System.Text;
-
using System.Windows.Forms;
-
using System.Runtime.Remoting;
-
using System.Runtime.Remoting.Channels;
-
using System.Runtime.Remoting.Channels.Tcp;
-
using Remoting;
-
-
namespace Client
-
{
-
public partial class ClientForm : Form
-
{
-
RemotingObject obj;
-
public ClientForm()
-
{
-
InitializeComponent();
-
StartClient();
-
}
-
-
private void 发送_Click(object sender, EventArgs e)
-
{
-
obj.TriggerEvent("客户端--" + this.ClientIP() + System.Environment.NewLine + textBox1.Text + System.Environment.NewLine);
-
}
-
//开启客户端
-
public void StartClient()
-
{
-
TcpChannel tcpchannel = new TcpChannel(0);
-
ChannelServices.RegisterChannel(tcpchannel, false);
-
-
obj = (RemotingObject)Activator.GetObject(typeof(RemotingObject), "tcp://localhost:8080/url");
-
-
}
-
//获取本地ip
-
public string ClientIP()
-
{
-
return System.Net.Dns.GetHostAddresses(System.Net.Dns.GetHostName())[0].ToString();
-
}
-
}
-
}
运行效果如下:
哈哈,如我所愿,利用事件确实实现了客户端与服务器的通信。
可是,这里面有一个小问题:在服务器订阅事件的时候我用到
RemotingObject.myEvent += new RemotingObject.MyDelegate(RemotingObject_myEvent);
这一句其实是我不得已而为之的,为什么这么说的。因为这里myEvent是静态的,只有这样我才可以用RemotingObject.myEvent订阅到事件。但是如果myEvent不是静态的呢?这时候如何去实例化一个对象,并且用"对象.myEvent"来订阅事件呢?如果我在服务器端自己new一个对象,那这个对象肯定和客户端激活得到的对象不是同一个引用。
(注:上面的Demo在此处下载http://download.csdn.net/detail/kkkkkxiaofei/5645629)
带着这个疑问,我找了些资料。有了重大发现。
[SecuritySafeCritical]
public static ObjRef Marshal(MarshalByRefObject Obj, string URI);
来看看官方的解释把:将给定的 System.MarshalByRefObject 转换为具有指定 URI 的 System.Runtime.Remoting.ObjRef 类的实例。
看到实例两个字我就激动了, 这里我的RemotingObject类就继承了System.MarshalByRefObject。这个ObjRef 类是做什么的?如果将其转换为ObjRef 类的实例又能怎么样呢?
再来看看ObjRef 吧。
ObjRef :存储生成代理以与远程对象通信所需要的所有信息。
有了上面的概念,我又有了点思路了:
之前客户端(RemotingObject)Activator.GetObject(typeof(RemotingObject), "tcp://localhost:8080/url")得到的这个对象,是服务器注册的RemotingConfiguration.RegisterWellKnownServiceType(typeof(RemotingObject),"url",WellKnownObjectMode.Singleton)这个对象的引用。在服务器端并不能获取这个对象。那如果我在服务器端显示创建一个对象,然后让客户端刚好能获取这个引用不就行了么。Marshal方法不就是显示生成服务器端的RemotingObject对象的代理么,客户端激活的代码已然不变,利用ip地址和url就能找到这个对象,这样2个程序域中的2个对象其实就是同一个引用。这么一来,就可以解决掉事件只能是静态的尴尬局面了,好了是时候验证我的想法了。代码如下,
远程对象:
-
using System;
-
using System.Collections.Generic;
-
using System.Linq;
-
using System.Text;
-
-
namespace Remoting
-
{
-
public class RemotingObject:MarshalByRefObject
-
{
-
public delegate void MyDelegate(string msg);
-
public event MyDelegate myEvent;
-
//触发事件
-
public void TriggerEvent(string msg)
-
{
-
if (myEvent != null)
-
myEvent(msg);
-
}
-
//无限生命周期
-
public override object InitializeLifetimeService()
-
{
-
return null;
-
}
-
}
-
}
客户端:
-
using System;
-
using System.Collections.Generic;
-
using System.ComponentModel;
-
using System.Data;
-
using System.Drawing;
-
using System.Linq;
-
using System.Text;
-
using System.Windows.Forms;
-
using System.Runtime.Remoting;
-
using System.Runtime.Remoting.Channels;
-
using System.Runtime.Remoting.Channels.Tcp;
-
using Remoting;
-
-
namespace Client
-
{
-
public partial class ClientForm : Form
-
{
-
RemotingObject obj;
-
public ClientForm()
-
{
-
InitializeComponent();
-
StartClient();
-
}
-
-
private void 发送_Click(object sender, EventArgs e)
-
{
-
obj.TriggerEvent("客户端--" + this.ClientIP() + System.Environment.NewLine + textBox1.Text + System.Environment.NewLine);
-
}
-
//开启客户端
-
public void StartClient()
-
{
-
TcpChannel tcpchannel = new TcpChannel(0);
-
ChannelServices.RegisterChannel(tcpchannel, false);
-
-
obj = (RemotingObject)Activator.GetObject(typeof(RemotingObject), "tcp://localhost:8080/url");
-
-
}
-
//获取本地ip
-
public string ClientIP()
-
{
-
return System.Net.Dns.GetHostAddresses(System.Net.Dns.GetHostName())[0].ToString();
-
}
-
}
-
}
服务器:
-
using System;
-
using System.Collections.Generic;
-
using System.ComponentModel;
-
using System.Data;
-
using System.Drawing;
-
using System.Linq;
-
using System.Text;
-
using System.Windows.Forms;
-
using System.Runtime.Remoting;
-
using System.Runtime.Remoting.Channels;
-
using System.Runtime.Remoting.Channels.Tcp;
-
using Remoting;
-
namespace Server
-
{
-
public partial class ServerForm : Form
-
{
-
RemotingObject marshal_obj;
-
public ServerForm()
-
{
-
InitializeComponent();
-
StartServer();
-
}
-
//开启服务器
-
public void StartServer()
-
{
-
TcpChannel tcpchannel = new TcpChannel(8080);
-
ChannelServices.RegisterChannel(tcpchannel,false);
-
-
//RemotingConfiguration.RegisterWellKnownServiceType(typeof(RemotingObject),"url",WellKnownObjectMode.Singleton);
-
-
marshal_obj = new RemotingObject();
-
ObjRef objRef = RemotingServices.Marshal(marshal_obj, "url");
-
marshal_obj.myEvent += new RemotingObject.MyDelegate(RemotingObject_myEvent);
-
}
-
-
void RemotingObject_myEvent(string msg)
-
{
-
//跨线程调用
-
textBox2.Invoke(new Action<string>(str => { textBox2.Text = str; }), msg);
-
}
-
}
-
}
运行结果很乐观,如期所料
看来用Marshal方法确实可以让服务器和客户端都有一份远程对象的引用,咦,那这么说,服务器岂不是也可以主动触发事件,然后让客户端来绑定,这样就能实现服务器向客户端通信。哈哈,离我的项目需求又近了一步,不错不错,下篇试试实现这个问题。
(注:上面的Demo在此处下载http://download.csdn.net/detail/kkkkkxiaofei/5646103)
初探Remoting双向通信(二)的更多相关文章
- 初探Remoting双向通信(四)
原 初探Remoting双向通信(四) 2013年06月26日 11:11:32 喜欢特别冷的冬天下着雪 阅读数 2632 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blo ...
- 初探Remoting双向通信(三)
原 初探Remoting双向通信(三) 2013年06月25日 17:51:08 喜欢特别冷的冬天下着雪 阅读数 4741 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blo ...
- 初探remoting双向通信(一)
原 初探remoting双向通信(一) 2013年06月24日 15:47:07 喜欢特别冷的冬天下着雪 阅读数 4389 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blo ...
- 初探JavaScript(二)——JS如何动态操控HTML
除去五一三天,我已经和<JavaScript Dom编程艺术>磨合了六天,第一印象很好.慢慢的,我发现这是一块排骨,除了肉还有骨头.遇到不解的地方就会多看几遍,实在不懂的先跳过,毕竟,初次 ...
- C# 实现Remoting双向通信
本篇文章主要介绍了C# 实现Remoting双向通信,.Net Remoting 是由客户端通过Remoting,访问通道以获得服务端对象,再通过代理解析为客户端对象来实现通信的 闲来无事想玩玩双向通 ...
- C# Remoting双向通信
闲来无事想玩玩双向通信,实现类似QQ的互发消息的功能.于是乎开始学习.Net Remoting. .Net Remoting 是由客户端通过Remoting,访问通道以获得服务端对象,再通过代理解析为 ...
- MongoDB初探系列之二:认识MongoDB提供的一些经常使用工具
在初探一中,我们已经能够顺利的将MongoDB在我们自己的机器上跑起来了. 可是在其bin文件夹以下另一些我们不熟知的工具.接下来,将介绍一下各个小工具的用途以及初探一中MongoDB在data文件夹 ...
- javascript --- 原型初探七日谈(二)
扩展内建对象: 在javascript中,内建对象的构造函数都是可以通过其原型来进行扩展的.这意味着我们可以做一些事情,例如我们要往数组原型中添加一个新方法,就可以在其所有的数组中使用,下面我们来试试 ...
- 跟我一起学WCF(2)——利用.NET Remoting技术开发分布式应用
一.引言 上一篇博文分享了消息队列(MSMQ)技术来实现分布式应用,在这篇博文继续分享下.NET平台下另一种分布式技术——.NET Remoting. 二..NET Remoting 介绍 2.1 . ...
随机推荐
- 对python中的__name__的理解
一开始学习python的时候,不理解python中的__name__的用途,一致感觉__name__的返回结果就是__main__ 今天系统的看了一下,才理解过来,__name__真正的用处是用在使用 ...
- 记录一次kibana启动Unable to fetch data from reporting collector
版本不匹配导致 应该es与kibana版本一致 本文链接:https://blog.csdn.net/qq_33293753/article/details/87894882
- Python之基本的日期与时间转换 datetime、 dateutil模块
简单举例datetime模块 from datetime import timedelta,datetime a = timedelta(days=2, hours=6) b = timedelta( ...
- git pull的时候发生冲突的解决方法之“error: Your local changes to the following files would be overwritten by merge”
今天在使用git pull 命令的时候发生了以下报错 目前git的报错提示已经相关友好了,可以直观的发现,这里可以通过commit的方式解决这个冲突问题,但还是想看看其他大佬是怎么解决这类问题的 在网 ...
- HDFS学习笔记一
一,HDFS是什么,HDFS有什么用,HDFS怎么用 HDFS:Hadoop Distribute File System 分布式文件系统 HDFS可以保证文件存储的可靠性, 二.HDFS的设计原则 ...
- Centos7下源码编译安装与配置redis5.0
1.下载redis5.0源码包 wget http://download.redis.io/releases/redis-5.0.5.tar.gz 2.检查是否安装过之前的历史版本 rpm -qa|g ...
- 离线安装.NET Framework 3.5
通过 NetFx3.cab 文件安装 NetFx3.cab下载链接: 链接:https://pan.baidu.com/s/1OPk-G_ChSwV89VZl6TV-LA 提取码:8zel 复制这段内 ...
- sql developer 中文乱码解决办法
近日在fedora13中安装了oracle和sql developer,在英文环境下启动sql developer正常,可是切换到中文环境下就显示乱码.google了一下,确定是因为JDK不支持中文的 ...
- Java 的内置对象
1.Request对象 该对象封装了用户提交的信息,通过调用该对象相应的方法可以获取封装的信息,即使用该对象可以获取用户提交的信息. 当Request对象获取客户提交的汉字字符时,会出现乱码问题 ...
- 饿了么监控系统 EMonitor 与美团点评 CAT 的对比
背景介绍 饿了么监控系统EMonitor:是一款服务于饿了么所有技术部门的一站式监控系统,覆盖了系统监控.容器监控.网络监控.中间件监控.业务监控.接入层监控以及前端监控的数据存储与查询.每日处理总数 ...