Castle 整合.NET Remoting

  今天研究了一下Castle的Remoting Facility.记录如下:

微软以前使用COM/DCOM的技术来处理分布式系统架构,通过Client端的Proxy代理程序来呼叫远程Server机器上的对象。.NET Framework则使用.NET Remoting或Web Services技术来实作分布式处理的工作概念;在这里针对.NET Remoting的设计架构做一个初步的简介和Castle整合示例。

.NET Framework提供了多种的机制来支持Remoting,如:

.利用Channel来负责信息的发送与接收。 .利用Formatter来负责在信息要通过channel发送出去之前,先将信息做适当的加密,或于信息在通过Channel接收进来之后,先将信息做相对的解密工作。 .利用Proxy来呼叫远程的对象执行所要的功能呼叫。

其关系如下图所示:

Channel 和 Formatter 在远程对象被使用之前,必须先在Server端注册好信息发送的信道(Channel),这些Channel可通过.NET Remotin configuration file或 ChannelServices对象类别的RegisterChannel方法来注册。

在Channel的使用上,.NET Framework支持HTTP、TCP及SMTP等通道。若使用HTTP Channel ,则使用SOAP协议来收送信息,所有的信息会被发送到SOAP Formatter中,被序列化(serialized)成XML的格式,而SOAP所需的headers也会被加入。至于使用TCP Channel者,则使用TCP协议来将信息发送到Binary Formatter中,以Binary Stream的方式来将信息发送到URI目的地。(URI : Universal Resource Identifier,类似大家所熟悉的URL)。

Activation and Proxy Server-Side Activation Server端在Client端要获取Remoting对象时必需在Server端能自动启动Remoting对象,可使用RemotingConfiguration对象类别的RegisterWellKnownServiceType方法来完成这项工作。

Client-Side Activation Client端要使用远程对象之前,可使用New 或Activator 对象类别所提供的CreateInstance或GetObject方法来启动对象并传回Proxy,以便Client端可通过Proxy来执行叫用远程对象的方法。

范例 以下分三个步骤来介绍

1. 建立Remoting对象

2. 在Server上初始Remoting物件

3. Client端使用Remoting对象

步骤1:建立Remoting对象 建立一个MathServer对象类别,提供Sum方法,可给予一连串的整数由Sum方法代为计算总和。程序代码如下,并说明于后:

using System;
namespace RemoteSample.Components
{
 /// <summary>
 /// Class1 的摘要说明。
 /// </summary>
 public interface IRemoteMath
 {
 int Sum(params int[] a);
 int CallCounter
 {
 get;
 }
 }
}
using System;
using RemoteSample.Components;
namespace RemoteSample.Components
{
 /// <summary>
 /// RemoteMath 的摘要说明。
 /// </summary>
 public class RemoteMath: MarshalByRefObject,IRemoteMath
 {
 private int callCounter = 0;
 public RemoteMath()
 {
 }
 #region 接口IRemoteMath的成员实现
 /// <summary>
 /// 求和计算
 /// </summary>
 /// <param name="a"></param>
 /// <returns></returns>
 public int Sum(params int[] a)
 {
 int sum = 0;
 for (int i = 0; i <= a.Length - 1; i++) 
 {
 sum += a[i];
 }
 callCounter += 1;
 return sum;
 }
 public int CallCounter
 {
 get
 {
 return this.callCounter;
 }
 }
 #endregion
 }
}

说明:Remoting对象必须继承自MarshalByRefObject,这样才能通过网络,将对象执行个体的参考位置传递给呼叫端。

步骤2:在Server上初始化Remoting对象,程序代码如下,并说明于后:

namespace RemoteSample.Server
{
 class RemoteServerMain
 {
 [STAThread]
 internal static void Main(string[] args)
 {
 IWindsorContainer container = new RemotingContainer();
 Console.ReadLine();
 }
 }
}

ServerConfig.xml文件:

<?xml version="1.0" encoding="utf-8" ?> 
<configuration>
 <facilities>
 <facility id="remote.facility" type="Castle.Facilities.Remoting.RemotingFacility, Castle.Facilities.Remoting"
 remotingConfigurationFile ="../../RemotingTcpConfig.config"
 isServer="true"
  registryUri="kernel.rem" >
  </facility>
 </facilities>
 <components>
 <component 
 id="remote.math" 
 service="RemoteSample.Components.IRemoteMath, RemoteSample.Components"
 type="RemoteSample.Components.RemoteMath, RemoteSample.Components"
 remoteserver="component" >
 </component>
 </components>
</configuration>
RemotingTcpConfig.config文件:
<?xml version="1.0" encoding="utf-8" ?>
<configuration>
 <system.runtime.remoting>
 <application>
 <channels>
 <channel ref="tcp" port="2133" />
 </channels>
 </application>
 </system.runtime.remoting>
</configuration>

说明:

使用Castle 的Remoting Facillity 使用Remoting 。

1.配置指出在2133 port上要建立TCP Channel, 2133 port上要建立tcp Channel

2.<components>指出在Server端注册所要使用的组件、服务的名称及启动的方式。其中component表示一个执行个体可供多个前端来呼叫,可保留其状态,另一种则为ClientActivated,一个执行个体只能服务一个前端的呼叫,无法保留其状态。

步骤3:在Client端使用Remoting对象

ClientConfig.xml
<?xml version="1.0" encoding="utf-8" ?> 
<configuration>
 <facilities>
 <facility 
 id="remote.facility" 
  type="Castle.Facilities.Remoting.RemotingFacility, Castle.Facilities.Remoting"
  remotingConfigurationFile="../../RemotingTcpConfigClient.config"
  isClient="true"
  remoteKernelUri="tcp://localhost:2133/kernel.rem"
  baseUri="tcp://localhost:2133" >
 </facility>
 </facilities>
 <components>
 <component 
 id="remote.math"
 service="RemoteSample.Components.IRemoteMath, RemoteSample.Components"
 type="RemoteSample.Components.RemoteMath, RemoteSample.Components"
 remoteclient="component" />
 </components>
</configuration>
RemotingTcpConfigClient.config
<?xml version="1.0" encoding="utf-8" ?>
<configuration>
 <system.runtime.remoting>
 <application>
 <channels>
 <channel ref="tcp" port="0" />
 </channels>
 </application>
 </system.runtime.remoting>
</configuration>

程序代码如下:

namespace RemoteSample.Client
{
 /// <summary>
 /// RemoteClient的摘要说明。
 /// </summary>
 public class RemoteClientMain
 {
 [STAThread]
 static void Main(String[] args)
 {
 IWindsorContainer container = new RemotingContainer();
 IRemoteMath remoteMath = (IRemoteMath)container[typeof(IRemoteMath)] ;
 Console.WriteLine("Client1 TCP Call Sum method {0} Counter {1}",remoteMath.Sum(10, 20, 30),remoteMath.CallCounter);
 Console.WriteLine("....press a key to stop");
 Console.ReadLine();
 }
 }
}

代码下载:RemotingSamples.rar

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏老码农专栏

TodoBackend展示应用以及ActFramework的实现

1305
来自专栏技术博客

Ioc模式和MEF

  分离关注( Separation of Concerns : SOC)是Ioc模式和AOP产生最原始动力,通过功能分解可得到关注点,这些关注可以是 组件Co...

842
来自专栏本立2道生

Win32对话框程序(1)

之前学C语言是一直都是在控制台下面操作的,面对的都是黑框框,严重的打击了学习的兴趣。后来在TC下进行C语言课程设计,做了图形界面编程,但都是点线面画的…… 

1771
来自专栏以南小隐-数通那些事儿

锐捷RSOS_10.4之后版本Ctrl层升级系统版本

1814
来自专栏微信终端开发团队的专栏

iOS微信特殊字符保护方案

一般来说,特殊字符闪退是系统漏洞引起,只要更新系统就行。但大部分用户不愿意更新系统...

81814
来自专栏武军超python专栏

2018年8月18日初识tkinter

把C盘里面的东西移动到其他盘对文件有影响吗?普通文件如音频视频没有影响,但是如果是软件的话 下载的时候会在注册表中记录打开文件的路径,如果移动到其他盘的话注册...

1102
来自专栏小灰灰

Java & PhantomJs 实现html输出图片

Java & PhantomJs 实现html输出图片 借助phantomJs来实现将html网页输出为图片 I. 背景 如何在小程序里面生成一张图,分享到朋...

7198
来自专栏点滴积累

Cesium中Clock控件及时间序列瓦片动态加载

前言 前面已经写了两篇博客介绍Cesium,一篇整体上简单介绍了Cesium如何上手,还有一篇介绍了如何将Cesium与分布式地理信息处理框架Geotrelli...

4544
来自专栏FreeBuf

技术分享:杂谈如何绕过WAF(Web应用防火墙)

0x01开场白 这个议题呢,主要是教大家一个思路,而不是把现成准备好的代码放给大家。 可能在大家眼中WAF(Web应用防火墙)就是“不要脸”的代名词。如果没有他...

2576
来自专栏菩提树下的杨过

更好用的excel国际化多语言导出

不知道大家在开发中有没有遇到过『excel导出』的需求,反正我最近写了不少这种功能,刚开始利用poi,一行行的手动塞数据,生成excel,而且还有国际化需求,比...

1392

扫码关注云+社区

领取腾讯云代金券