一、IInterceptingProxyFactory
对于Dora.Interception来说,方法调用之所有能够被拦截的根源在于我们改变了服务实例的提供方式,原来的对象被替换成了可被拦截的代理对象。针对代理对象的提供体现在如下这个IInterceptingProxyFactory接口上。如果提供类型体现为一个接口,Wrap方法会被调用来创建一个封装目标对象的代理(如果不需要被拦截,则直接返回目标对象);如果提供类型体现为一个类型,Create方法则被用来实现对代理对象的创建,如果不需要被拦截,方法提供的后面两个参数会被用来提供目标对象。
public interface IInterceptingProxyFactory { IServiceProvider ServiceProvider { get; } object Wrap(Type typeToIntercept, object target); object Create(Type typeToIntercept, IServiceProvider serviceProvider, Func<object> targetAccessor = null); }
二、InterceptingProxyFactoryBase
Dora.Interception提供了如下一个实现了IInterceptingProxyFactory接口的基类InterceptingProxyFactoryBase。后者帮助我们实现针对拦截器的解析,解析后的拦截器体现为一个InterceptorDecoration对象。作为它的派生类型只需要实现两个受保护的虚方法Wrap和Create根据解析出来的拦截器实现可被拦截的代理对象的创建。
public abstract class InterceptingProxyFactoryBase : IInterceptingProxyFactory { public IInterceptorResolver InterceptorResolver { get; } public IServiceProvider ServiceProvider { get; } public InterceptingProxyFactoryBase(IInterceptorResolver interceptorResolver, IServiceProvider serviceProvider); public object Create(Type typeToIntercept, IServiceProvider serviceProvider, Func<object> targetAccessor = null); public object Wrap(Type typeToIntercept, object target); protected virtual bool CanIntercept(Type typeToIntercept); protected abstract object Wrap(Type typeToIntercept, object target, InterceptorDecoration interceptors); protected abstract object Create(Type typeToIntercept, IServiceProvider serviceProvider, InterceptorDecoration interceptors); }
如果所示的是InterceptorDecoration, 我们可以得到应用到目标类型中所有方法(包括属性的Get和Set方法)上的拦截器(实际上所有拦截器按照指定顺序构建而成的拦截器管道,最终体现为一个类型为InterceptorDelegate 的委托对象)。
public sealed class InterceptorDecoration { public InterceptorDelegate GetInterceptor(MethodInfo methodInfo); public MethodInfo GetTargetMethod(MethodInfo methodInfo); public bool IsInterceptable(MethodInfo methodInfo); public IReadOnlyDictionary<int, InterceptorDelegate> Interceptors { get; } public bool IsEmpty { get; } public IReadOnlyDictionary<MethodInfo, MethodBasedInterceptorDecoration> MethodBasedInterceptors { get; } public IReadOnlyDictionary<PropertyInfo, PropertyBasedInterceptorDecoration> PropertyBasedInterceptors { get; } } public class MethodBasedInterceptorDecoration { public InterceptorDelegate Interceptor { get; } public MethodInfo Method { get; } } public class PropertyBasedInterceptorDecoration { public PropertyInfo Property { get; } public MethodBasedInterceptorDecoration GetMethodBasedInterceptor { get; } public MethodBasedInterceptorDecoration SetMethodBasedInterceptor { get; } }
自定义的IInterceptingProxyFactory实现只需要按照普通的服进行注册即可。
三、针对Castle的集成
由于Castle原生的框架并没有提供针对Task的支持,所以我们利用另一个名为Castle.Core.AsyncInterceptor将Castle的拦截实现整合到Dora.Interception。具体的实现体现在如下这个DynamicProxyFactory中。该类型对应的NuGet包为“Dora.Interception.Castle”。
public class DynamicProxyFactory : InterceptingProxyFactoryBase { public DynamicProxyFactory(IInterceptorResolver interceptorResolver, IServiceProvider serviceProvider); protected override object Create(Type typeToIntercept, IServiceProvider serviceProvider, InterceptorDecoration interceptors); protected override object Wrap(Type typeToIntercept, object target, InterceptorDecoration interceptors); }
如果需要采用基于Caslte的拦截实现机制,我们只需要做如下的设置即可。
public class Startup { public void ConfigureServices(IServiceCollection services) { services ... AddInterception(builder=>builder.SetCastleDynamicProxy()); } ... }
或者
public class Startup { public IServiceProvider ConfigureServices(IServiceCollection services) { return services ... .BuildInterceptableServiceProvider(builder=>builder.SetCastleDynamicProxy()); } ... }
[1]:更加简练的编程体验
[2]:基于约定的拦截器定义方式
[3]:多样性的拦截器应用方式
[4]:与依赖注入框架的深度整合
[5]:对拦截机制的灵活定制
微信公众账号:大内老A
微博:www.weibo.com/artech
如果你想及时得到个人撰写文章以及著作的消息推送,或者想看看个人推荐的技术资料,可以扫描左边二维码(或者长按识别二维码)关注个人公众号)。
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。