Remoting IPC 的几种Channel
使用Remoting来实现进程间通信是会面临的一个问题是:选择哪一种信道作为数据传输的通道?
首先来看看有哪几种信道,以下的截图是System.Runtime.Remoting程序集的结构:
从中可以看到在channels文件夹下有http、tcp、ipc三个文件夹,分别对应了三种信道,这三种信道分别对应了不同的数据通信方式。
IpcChannel
通常情况下,在同一台机器上的进程间通信,推荐使用IpcChannel,在这种情况下IpcChannel能够提供最高的性能,因为其底层使用了命名管道,而命名管道的底层是共享内存。显然,这样直接的内存共享要比其它各种协议的数据传输效率来得高。通常的使用方法如下:
public static void RegisterChannel(string portName)
{
//使用二进制作为数据序列化方式,也可以使用SoapClientFormatterSink、SoapServerFormatterSink,如果点开上图中的sinks文件夹,会看到binary和soap的格式化器源码文件
var serverProvider = new BinaryServerFormatterSinkProvider { TypeFilterLevel = TypeFilterLevel.Full };
var clientProvider = new BinaryClientFormatterSinkProvider();
var properties = new Hashtable();
//指定信道的端口
properties["portName"] = "client";
var channel = new IpcChannel(properties, clientProvider, serverProvider);
//向系统的信道服务注册信道
ChannelServices.RegisterChannel(channel, false);
}
通过上面的方法就在当前进程(A进程)中完成了信道的注册,注册完信道后就可以往信道中丢东西了:
var remoteObject = new RemoteObject();
RemotingServices.Marshal(remoteObject, "RemoteObject", typeof(RemoteObject));
需要注意的是RemoteObject一定是可以跨越进程边界的,需要是MarshalByRefObject的子类。
此时在其他进程(B进程)就可以通过信道访问到刚刚共享到信道中的实例了:
var url = "ipc://" + "client" + "/RemoteObject";
var remoteObject = (RemoteObject)Activator.GetObject(typeof(RemoteObject), url);
如果需要在B进程中往A进程发送消息也非常简单,直接利用事件回调,在B进程中触发remoteObject的事件,将消息广播到A进程。
TcpChannel
TcpChannel使用二进制格式化将数据序列化为二进制流,并使用TCP协议在进程间传输数据。其使用方法跟IpcChannel类似,就不贴代码了,但是需要注意,使用中一定要保证协议的一致,不能使用http去获取tcp的数据。通常在跨机器的远程调用中使用TcpChannel。
HttpChannel
HttpChannel使用soap协议来传输数据,所有的数据在传递之前都会经过一个叫做SoapFormatter的格式化器格式化为xml,使用方式跟IpcChannel类似,也不贴代码了。HttpChannel一次只向给定服务器打开指定数目的连接,默认值为 2,但您可以使用clientConnectionLimit来更改该默认值。在BS的通信中,推荐使用HttpChannel。