如果说到拦截器,相信大家都不陌生,所有的AOP都依赖各种形式的拦截器。为了让WCF服务端的行为发生变化,这里要借助拦截器的力量。

问题:WCF里面如何创建一个拦截器?

    事实上,WCF里面有很多种拦截器,分别用于拦截不同的信息。这里要改变的是方法的实现部分,因此,只要拦截WCF操作就可以达到目的。

    那么,如何拦截操作哪?

    如果熟悉WCF的话,那么,一定知道有这么一个接口:IOperationInvoker

    这个接口的核心方法为:Invoke及其异步方法

问题:如何把一个实现IOperationInvoker的实例注入WCF

    仔细看一下MSDN,或者看reflector,就可以发现,所有实现IOperationInvoker的类型,几乎都是通过各种OperationBehavior加入的。

    而所有的OperationBehavior都实现了一个IOperationBehavior接口。

主角和设计约束

    主角们在刚才的两个问题中已经全部登场了:IOperationInvokerIOperationBehavior

    剩下来的问题是如何实现这两位主角。

    为了简化期间,这里只考虑这样的情况:所有的操作只有一个输入值和一个输出值,以及只有同步操作。当然这里的一个输入/输出是指一个简单或复杂值,也就说,需要传多个值时,使用一个自定义类型来包裹这多个值。

实现

    先说说实现IOperationInvoker,这里盗用一下MVC的概念,不妨将我们的实现控制器声明为:ControllerInvoker

    简单的示意:

   1:      internal sealed class ControllerInvoker
   2:          : IOperationInvoker
   3:      {
   4:   
   5:          private readonly IOperationInvoker Inner;
   6:   
   7:          public ControllerInvoker(IOperationInvoker inner)
   8:          {
   9:              Inner = inner;
  10:          }
  11:   
  12:          public object[] AllocateInputs()
  13:          {
  14:              return Inner.AllocateInputs();
  15:          }
  16:   
  17:          public object Invoke(object instance, object[] inputs, out object[] outputs)
  18:          {
  19:              // do something before invoking
  20:              object result = Invoke(instance, inputs, out outputs);
  21:              // do something after invoking
  22:              return result;
  23:          }
  24:   
  25:          public IAsyncResult InvokeBegin(object instance, object[] inputs, AsyncCallback callback, object state)
  26:          {
  27:              throw new NotSupportedException();
  28:          }
  29:   
  30:          public object InvokeEnd(object instance, out object[] outputs, IAsyncResult result)
  31:          {
  32:              throw new NotSupportedException();
  33:          }
  34:   
  35:          public bool IsSynchronous
  36:          {
  37:              get { return true; }
  38:          }
  39:      }

     我们的Invoker目前什么好事情也没做(坏事倒是做了一桩,引入了不支持异步),别急,这个类还没完工哪。

    再看看Behavior需要做什么:

   1:      [AttributeUsage(AttributeTargets.Method, Inherited = false, AllowMultiple = false)]
   2:      public sealed class ControllerAttribute
   3:          : Attribute, IOperationBehavior
   4:      {
   5:   
   6:          void IOperationBehavior.AddBindingParameters(OperationDescription operationDescription, BindingParameterCollection bindingParameters) { }
   7:   
   8:          void IOperationBehavior.ApplyClientBehavior(OperationDescription operationDescription, ClientOperation clientOperation) { }
   9:   
  10:          void IOperationBehavior.ApplyDispatchBehavior(OperationDescription operationDescription, DispatchOperation dispatchOperation)
  11:          {
  12:              dispatchOperation.Invoker = new ControllerInvoker(dispatchOperation.Invoker);
  13:          }
  14:   
  15:          void IOperationBehavior.Validate(OperationDescription operationDescription) { }
  16:   
  17:      }

    好,拦截的外壳已经准备好了,来一个实例看看:

   1:      [ServiceContract(Namespace="urn:Zhenway.Test")]
   2:      public interface ITest
   3:      {
   4:          [Controller]
   5:          [OperationContract]
   6:          TestResponse Test(TestRequest req);
   7:      }

    是不是很简单。

下篇预告

    今天说了半天,都在说如何拦截的问题,还有个更大的问题如何路由,以及怎么实现动态路由,请看下篇。

posted on 2010-10-27 14:01  Zhenway  阅读(4908)  评论(1编辑  收藏  举报