微软的远程处理框架.NET Remoting - 2
连接在一起是如此的简单。以下的服务器应用提供了一个服务,可将一个字符串的字母顺序反转。
server.cs using system;
using system.io;
using system.runtime.remoting;
using system.runtime.remoting.channels.http;
namespace remotingsample
{
public class reverser : marshalbyrefobject
{
public string reverse(string text)
{
console.writeline("reverse({0})", text);
string rev = "";
for (int i=text.length-1; i>=0; i--)
{
rev += text[i];
}
console.writeline("returning : {0}", rev);
return rev;
}
}
public class theapp
{
public static void main()
{
file:// create a new http channel that
// listens on port 8000
httpchannel channel = new httpchannel(8000);
// register the channel with the runtime
channelservices.registerchannel(channel);
// expose the reverser object from this server
remotingservices.registerwellknowntype(
"server", // assembly name
"remotingsample.reverser", // full type name
"reverser.soap", file:// uri
wellknownobjectmode.singleton // instancing mode
);
// keep the server running until
// the user presses enter
console.writeline("server.exe");
console.writeline("press enter to stop server...");
console.readline();
}
}
}
现在我们已经拥有了一个字符反向服务,以下我们将建立一个客户应用来使用这个服务:
client.cs using system;
using system.runtime.remoting;
using system.runtime.remoting.channels.http;
using remotingsample; // reference the server
public class theapp
{
public static void main()
{
// create and register a channel
// to comunicate to the server.
// the client will use port 8001
// to listen for callbacks
httpchannel channel = new httpchannel(8001);
channelservices.registerchannel(channel);
// create an instance on the remote server
// and call a method remotely
reverser rev = (reverser)activator.getobject(
typeof(reverser), // type to create
"http://localhost:8000/reverser.soap" file:// uri
);
console.writeline("client.exe");
console.writeline(rev.reverse("hello, world!"));
}
}
看,通过远程.net将两个应用连接在一起是多么的简单。当服务端和客户端程序放在两台不同的机器时,我们可以令两个程序都运行在80端口。这样远程的调用就可通过一个防火墙。你也可将httpchannel改为一个tcpchannel试一下。
你要注意到,客户端是通过“reverser.soap”来标识它想连接的对象的。这个名字与服务器代码中registerwellknowntype的uri参数符合。“.soap”的扩展是不必要的。uri可以是任何的字符串,只要它能唯一标识服务器的对象就可以了。“.soap”的扩展只是用来提醒我们http channel是使用soap来格式化信息的。
在上面有关channel的例子中,你可能会产生这样的疑问:参数是如何跨网络传送,返回值又是如何送回的呢?答案是,在参数被跨网络传送之前,他们必须经过串行化处理。对于需要传送的所有对象或者结构,都要经过这样的处理。串行化的处理很简单,只是以连续字节的方式建立变量或者对象中的数据的一个持续拷贝。将这些字节还原为一个值或者对象实例的处理被称为反串行化。
那么参数是如何串行化的呢?远程.net架构为我们提供了一个称为格式器(formatters)的对象集。格式器可将一个对象变成是一个特定的持续数据格式,也可以将该它还原回来。.net为我们提供了两种格式器:
system.runtime.serialization.formatters.binary
system.runtime.serialization.formatters.soap
binary(二进制)格式器是最简单的。它只是将数据直接转换为一个字节流。soap格式器使用一个xml来保持一个对象数据。要知道soap更详细的信息,可到http://www.soapwebservices.com。
以下我们举一个有关格式器的简单例子。我们将使用soap格式器,由于它使用的是xml,我们可以很容易地读出串行化的数据。
soap.cs using system;
using system.io;
using system.runtime.serialization.formatters.soap;
public class person
{
public string firstname = "david";
public string lastname = "findley";
private int age = 29;
}
public class theapp
{
public static void main()
{
stream stream = file.create("example.xml");
soapformatter formatter = new soapformatter();
person p = new person();
// persist an integer
formatter.serialize(stream, 5);
file:// persist a string
formatter.serialize(stream, "this is a string");
// persist an object
formatter.serialize(stream, p);
stream.close();
}
}
对于每个串行化的调用,example.xml的内容将有三个不同的部分:
example.xml
<soap-env:body>
<xsd:int id="ref-1">
<m_value>5</m_value>
</xsd:int>
</soap-env:body>
<soap-env:body>
<soap-enc:string id="ref-1">this is a string</soap-enc:string>
</soap-env:body>
<soap-env:body>
<a1:person id="ref-1">
<firstname id="ref-3">david</firstname>
<lastname id="ref-4">findley</lastname>
<age>29</age>
</a1:person>
</soap-env:body>
你可以看出,它可以串行化基本值类和对象。httpchannel使用soap格式器在客户和服务器之间传送数据。
总的来说,格式器可以格式和保持值或者对象的数据。channel传送和接收数据。通过channel和格式器的协同工作,我们将可以使用任何的网络和协议来连接两个应用。

浙公网安备 33010602011771号