Remoting是NET平台下比较成熟高效的分布式技术,我们习惯采用传统的远程调用的方式使用Remoting。在客户端所在的Application Domain,我们通过Proxy(Transparent Proxy)远程地跨Application Domain调用一个方法。当来自Client端的调用请求通过Proxy到达Server端所在的Application Domain后,Remoting Infrastructure在Server 端激活(Activate)相应的远程对象(一个继承子System.MarshalByRefObject类对象)——这里仅仅以服务端激活对象(Server Activated Object——SAO),然后再Server端执行相应的操作后把Result传递给Proxy,并最终到达Client。这是一种典型的Request/Response的调用方式。
我之所以一直比较推崇在.NET平台下使用Remoting而非XML Web Service是因为我觉得.NET Remoting是一种比较成熟的分布式技术。它自身提供了XML Web Service很多不具备的特性,其中对双向通信的支持就是一个很好的体现。
相对于典型的Request/Response的消息交换模式(Message Exchange Pattern——MEP),双向通信实质上是采用的Duplex的MEP。也就是说,Server端在执行操作的时候,可以回调(Callback)Client端的操作(这个操作时再Client端的Application Domain中执行的)。
现在我们来看如何一步一步实现在Remoting环境下的双向通信。在下面的Sample中,我们的逻辑是:调用一个数学计算的远程调用,除了传递相应的操作数之外,我们还传递一个对象,这个对象可以在Server端中回调 (Callback) 把运算结果在Client端显示出来。可以通过下面的URL下载源代码:http://www.cnblogs.com/files/artech/Artech.DuplexRemoting.zip
步骤一、构建整个Solution的整体构架
如果没有把远程对象的Interface,对已某一个需要调用这个远程对象的Client来说,它必须引用远程对象本身。从安全的角度考虑,Server向Client过多暴露了操作的实现逻辑。如果我们把远程操作的Contract提取出来,Client只要引用这个Interface就可以了。
一般来说,远程对象的Contract相对时静态的(static),而业务逻辑的实现则是经常 变化的。因为Client只需要了解的是远程对象的Contract,所在无论Server端对远程对象的实现作了多大的变动,对不回对Client产生任何影响。
IDuplexCalculator
1: public interface IDuplexCalculator
2: {
3: void Add(double x, double y, ICalculatorCallback callback);
4: }
ICalculatorCallback
1: public interface ICalculatorCallback
2: {
3: void ShowResult(double x, double y, double result);
4: }
DuplexCalculatorRemoting
1: public class DuplexCalculatorRemoting : MarshalByRefObject, IDuplexCalculator
2: {
3: public void Add(double x, double y, ICalculatorCallback callback)
4: {
5: Console.WriteLine("Invoke the method Add({0},{1}).", x, y);
6: double result = x + y;
7: callback.ShowResult(x, y, result);
8: }
9: }
App.config
1: <configuration>
2: <system.runtime.remoting>
3: <application name="Calculator">
4: <service>
5: <wellknown mode="SingleCall"
6: type="Artech.DuplexRemoting.Remoting.DuplexCalculatorRemoting,Artech.DuplexRemoting.Remoting"
7: objectUri="DuplexCalculator.soap" />
8: </service>
9:
10: <channels>
11: <channel ref="http" port="8080">
12: <serverProviders>
13: <provider ref="wsdl" />
14: <formatter ref="binary" typeFilterLevel="Full" />
15: </serverProviders>
16: <clientProviders>
17: <formatter ref="binary" />
18: </clientProviders>
19: </channel>
20: </channels>
21: </application>
22: </system.runtime.remoting>
23: </configuration>
Program.cs
1: class Program
2: {
3: static void Main(string[] args)
4: {
5: System.Runtime.Remoting.RemotingConfiguration.Configure("Artech.DuplexRemoting.Hosting.exe.config",false);
6: Console.WriteLine("Calculator service has begun to listen ");
7: Console.Read();
8: }
9: }
这里需要特别注意的有以下两点:
之所以要把typeFilterLevel为Full,是因为我们的远程调用里包含一Callback对象,它实际上是一个继承System.MarshalByRefObject类对象(这个的对象将在Artech.DuplexRemoting.Client中定义)。而这个对象是不会再Low Level下被自动反序列化。
1: <channels>
2: <channel ref="http" port="8080">
3: <serverProviders>
4: <provider ref="wsdl" />
5: <formatter ref="binary" typeFilterLevel="Full" />
6: </serverProviders>
7: <clientProviders>
8: <formatter ref="binary" />
9: </clientProviders>
10: </channel>
11: </channels>
1: public interface IDuplexCalculator
2: {
3: void Add(double x, double y, ICalculatorCallback callback);
4: }
CalculatorCallbackHandler
1: public class CalculatorCallbackHandler : MarshalByRefObject, ICalculatorCallback
2: {
3: public void ShowResult(double x, double y, double result)
4: {
5: Console.WriteLine("x + y = {2} where x = {0} and y = {1}", x, y, result);
6: }
7: }
App.config
1: <configuration>
2: <system.runtime.remoting>
3: <application>
4: <channels>
5: <channel ref="http" port="0">
6: <clientProviders>
7: <formatter ref="binary" />
8: </clientProviders>
9: <serverProviders>
10: <formatter ref="binary" typeFilterLevel="Full" />
11: </serverProviders>
12: </channel>
13: </channels>
14: </application>
15: </system.runtime.remoting>
16: </configuration>
Program.cs
1: class Program
2: {
3: static void Main(string[] args)
4: {
5: System.Runtime.Remoting.RemotingConfiguration.Configure("Artech.DuplexRemoting.Client.exe.config", false);
6:
7: InvocateDuplexCalculator("http://localhost:8080/Calculator/DuplexCalculator.soap");
8: }
9:
10: static void InvocateDuplexCalculator(string remoteAddress)
11: {
12: IDuplexCalculator proxy = (IDuplexCalculator)Activator.GetObject(typeof(IDuplexCalculator), remoteAddress);
13: proxy.Add(1, 2, new CalculatorCallbackHandler());
14: Console.Read();
15: }
16: }
这里有两点需特别注意的:
到现在为止我们已经完成了所有的Program,我们来运行一下,在客户端你将得到如下的输出:
1: x + y = 3 where x = 1 and y = 2
步骤六、将远程对象Host到IIS中
我们知道,Remoting有两种Host方式Self Host和IIS Host,上面我们把Remoting Host到一个Console Application中; 现在我们把试着把它Host到IIS中。实际上我们要做的工作很简单。
1: <configuration xmlns="http://schemas.microsoft.com/.NetConfiguration/v2.0">
2: <system.runtime.remoting>
3: <application>
4: <service>
5: <wellknown mode="SingleCall"
6: type="Artech.DuplexRemoting.Remoting.DuplexCalculatorRemoting,Artech.DuplexRemoting.Remoting"
7: objectUri="DuplexCalculator.soap" />
8: </service>
9: <channels>
10: <channel ref="http">
11: <serverProviders>
12: <provider ref="wsdl" />
13: <formatter ref="binary" typeFilterLevel="Full" />
14: </serverProviders>
15: <clientProviders>
16: <formatter ref="binary" />
17: </clientProviders>
18: </channel>
19: </channels>
20: </application>
21: </system.runtime.remoting>
22: </configuration>
这样我们可以不需要Hosting,就可以运行Client了。
我所理解的.NET Remoting:
我所理解的Remoting(1):Marshaling & Activation - Part I
我所理解的Remoting(1):Marshaling & Activation - Part II
我所理解的Remoting(2):远程对象生命周期的管理—Part I
我所理解的Remoting(2):远程对象的生命周期管理-Part II
我所理解的Remoting(3):创建CAO Service Factory使接口和实现相互分离