使用dynamic和MEF实现轻量级的AOP组件 (3)
转摘 https://www.cnblogs.com/niceWk/archive/2010/07/22/1783068.html
水到渠成
在上一篇的《偷梁换柱》中,介绍了WeavableObject的基本实现,本篇将继续进一步探讨它的更多细节。
首先我们来看一下方法拦截点(AOP术语称为joinpoint 加入点)的位置,通常的AOP对方法的拦截有3种,一种是Before method call, 它设置在进入原方法体之前,一种是After method call,它设置在方法返回之前,还有一种就是Around,它的行为很霸道,就是不执行原方法。后来基于不同的语言特性,有些平台开始出现Before return、 After return、Before throw exception等等,通过仔细的衡量,笔者决定在一个方法中设置如下拦截点,即能保证满足大部分功能需要,又能保证结构上的简单:
public override bool TryInvokeMember(InvokeMemberBinder binder, object[] args, out object result)
{
result = null;
var mi = typeof(T).GetMethods()
.FirstOrDefault(m => m.Name == binder.Name
&& binder.ReturnType.IsAssignableFrom(m.ReturnType)
&& IsMatchParameters(m.GetParameters(), args));
if (mi != null)
{
WeavingContext context = new WeavingContext(source, binder, args);
try
{
// Before method call 拦截点
//Around method call 拦截点
//调用原方法
context.ReturnValue = mi.Invoke(source, args);
}
catch (Exception ex)
{
//异常处理拦截点
ProcessException(context, ex);
}
finally
{
// After method call 拦截点
result = context.ReturnValue;
}
return true;
}
else
{
return false;
}
}
对象属性get/set的拦截类似,大家自己可以看代码,这里从略。
解决了拦截点的定义,下一步就是如何织入拦截器方法(AOP术语叫做Advice)。初看貌似可以使用事件,也就是说在不同的拦截点触发相应的事件以此触发预先注册的拦截类的事件处理器,但使用事件有一个问题无法解决,那就是如果有多个事件处理器,我们很难保证事件处理器的执行次序。为此,特意设计了一个对象链类AspectChain(从LinkedList<T>派生),为什么不是List<T>? 主要的原因是我们需要从两头沿着节点来调用链中的对象,基本的调用原则是:
Before method call : 从链头到链尾;
Around method call: 从链头到链尾;
Exception method call:从链头到链尾;
After method call: 从链尾到链头。
AspectChain采用责任链设计模式,其中有两个方法值得注意,第一个方法是HasAroundMethod,如果链中只要任何一个对象携带有对AroundMethodCall的方法,该方法返回true,否则返回false。因为Around注入的方法和被注入的原方法二者是相互排斥的,它们之中只能有一个被执行,如果有Around方法的话,那么Around方法就要替代原方法。第二个方法是ExceptionMethodCall,我们这样来处理异常,也就是如果链中有一个对象的异常处理方法返回true,就表示异常事件不再继续传递,异常处理完毕,否则一直传递到最后一个节点,如果所有的节点都返回false,那么异常将被抛出到外部对象去处理。
现在我们对如何设计Aspect的结构的思路逐渐明朗化了,那就是每个Aspect都必须携带可以被拦截点回调的方法,实际的回调是发生在AspectChain链中的。
下面就是我们设计的IAspect接口:
public interface IAspect
{
void OnBeforeMethodCall(WeavingContext context);
bool HasArroundMethod(WeavingContext context);
void OnAroundMethodCall(WeavingContext context);
void OnAfterMethodCall(WeavingContext context);
bool OnExceptionMethodCall(WeavingContext context, Exception ex);
void OnPropertyChanging(WeavingContext context);
void OnPropertyChanged(WeavingContext context);
void OnBeforeGetValue(WeavingContext context);
void OnAfterGetValue(WeavingContext context);
List<string> Targets { get; }
bool IsMatch(string target);
int Sequence { get; set; }
bool Enabled { get; set; }
}
并不像是每个Aspect对象都要实现所有接口中的方法,所以我们就很有必要为这些对象设计一个共同的抽象基类AspectBase,这样做的好处能够从DynamicAspect的设计中看出来。
接口还包含的几个属性和一个IsMatch的方法。在上一篇要处理的问题列表中的问题6, 我们解决了前一半,那么Aspect如何匹配目标对象的问题,我们需要一种方法来辨别目标对象,笔者采用正则表达式将目标对象的类型名称和正则表达式的模板进行匹配,一个Aspect可以匹配多个目标对象,也就是我们可以在Aspect对象中指定多个模板,模板保存在Targets属性中,这样我们只要遍历这个链表就可以获取匹配的信息。IsMatch的实现在AspectBase中实现,注意IsMatch在基类被定义为虚拟的,也就是说它可以被派生类重载,允许派生类实现它们自己的匹配规则。Sequence属性用来告知对象被插入到AspectChain的次序,Enabled则作为一个开关,如果该值被设置为false, 那么这个Aspect将不再参与编织。默认值是true。下面是IsMatch接口方法在AspectBase中的实现:
public virtual bool IsMatch(string target)
{
if (Targets.Count == 0)
return true; return Targets.Any(t => Regex.IsMatch(target, t));
}
如果没有模板存在在Target表中,则认为匹配所有的目标对象类型。为了方便正则表达式模板的编写,特意设计一个助手类TargetPattern,但这个类并不一定需要,它的一个主要目的就是减少一些硬编码。比如你可以用TargetPattern.All 来代替 @“\w+?"。我们解决了目标对象的匹配问题,采用同样的方法,我们可以在重载的IAspect实现方法中让具体的Aspect类来决定如何匹配目标方法。大家可以下载包里面的例子看到,在例子中使用的是硬编码,但是具体Aspect类的实现不是DynamicAspect的任务(除了一些广泛通用的Aspect之外,DynamicAspect的未来任务就是内建对通用Aspect的实现)。每个Aspect的开发者都可以自己实现对方法的匹配算法,问题是每个具体的Aspect是如何知道调用方的信息呢?答案就在作为方法参数传递的WeavingContext对象,那么就让我们看一下这个对象都有什么成员来提供必要的信息:
public class WeavingContext
{
private DynamicMetaObjectBinder binder; public WeavingContext(object target, DynamicMetaObjectBinder binder, object[] argumentValues=null, object returnValue=null)
{
this.Target = target;
this.binder = binder;
this.ArgumentValues = argumentValues;
this.ReturnValue = returnValue;
} public object[] ArgumentValues { get; set; } public object ReturnValue { get; set; } public object Target { get; private set; }
public InvokeMemberBinder InvokeMemberBinder
{
get { return binder as InvokeMemberBinder; }
} public InvokeBinder InvokeBinder
{
get { return binder as InvokeBinder; }
} public GetIndexBinder GetIndexBinder
{
get { return binder as GetIndexBinder; }
}
public SetIndexBinder SetIndexBinder
{
get { return binder as SetIndexBinder; }
}
public GetMemberBinder GetMemberBinder
{
get { return binder as GetMemberBinder; }
}
public SetMemberBinder SetMemberBinder
{
get { return binder as SetMemberBinder; }
}
WeavingContext类包含四个重要的信息,即使用构造器参数传入对象的四个对象:
Target是目标对象实例,该对象可作为Aspect计算匹配的信息来源;Binder对象,Binder对象含有调用方绑定到动态方法上的元数据信息,如被调用的方法名称,参数个数等,不同的动态方法有不同的Binder类型,为了避免在代码中写很多if elseif 这样的语句,我们将不同类型的Binder做成属性,以方便在Aspect的方法中调用。ArgumentValues顾名思义就是参数值集合,同样ReturnValue则用于保存方法的返回值(如果有的话)。
我想我们已经解决了要解决的7个问题中的6个,剩下问题7将在下一篇当我们揭示实现DynamicAspect的另一半MEF (Managed Extensibility Framework)时予以阐述。
(未完待续)
使用dynamic和MEF实现轻量级的AOP组件 (3)的更多相关文章
- 使用dynamic和MEF实现轻量级的AOP组件 ---- 系列文章
.NET 4 实践 - 使用dynamic 和MEF实现轻量级的AOP组件(1) .NET 4 实践 - 使用dynamic和MEF实现轻量级的AOP组件 (2) .NET 4 实践 - 使用 ...
- 使用dynamic和MEF实现轻量级的AOP组件 (2)
转摘 https://www.cnblogs.com/niceWk/archive/2010/07/21/1782092.html 偷梁换柱 上一篇我们初试了DynamicAspect这把小刀,如果你 ...
- 使用dynamic 和MEF实现轻量级的 AOP 组件 (1)
转载https://www.cnblogs.com/niceWk/archive/2010/07/19/1780843.html AOP魔法 今天你AOP了吗?谈到AOP,总有一种神秘的感觉,人类对于 ...
- .NET 4 实践 - 使用dynamic和MEF实现轻量级的AOP组件 (4)
转摘 https://www.cnblogs.com/niceWk/archive/2010/07/23/1783394.html 借花献佛 前面我们介绍了构成DynamicAspect绝大部分的类, ...
- C++11实现一个轻量级的AOP框架
AOP介绍 AOP(Aspect-Oriented Programming,面向方面编程),可以解决面向对象编程中的一些问题,是OOP的一种有益补充.面向对象编程中的继承是一种从上而下的关系,不适合定 ...
- C#轻量级高性能日志组件EasyLogger
一.课程介绍 本次分享课程属于<C#高级编程实战技能开发宝典课程系列>中的第六部分,阿笨后续会计划将实际项目中的一些比较实用的关于C#高级编程的技巧分享出来给大家进行学习,不断的收集.整理 ...
- 基于DispatchProxy打造自定义AOP组件
DispatchProxy是微软爸爸编写的一个代理类,基于这个,我扩展了一个AOP组件 暂时不支持依赖注入构造方法,感觉属性注入略显麻烦,暂时没打算支持 基于特性的注入流程 [AttributeUsa ...
- 使用MEF与Castle实现AOP
MEF是微软的一个ioc框架,使用非常方便,我们只需要在需要导出的类上标记[Export],在需要使用的地方[import]就可以使用了.现在我们扩展MEF,在其装配生成实例时,使用Castle Dy ...
- 基于微软企业库的AOP组件(含源码)
软件开发,离不开对日志的操作.日志可以帮助我们查找和检测问题,比较传统的日志是在方法执行前或后,手动调用日志代码保存.但自从AOP出现后,我们就可以避免这种繁琐但又必须要实现的方式.本文是在微软企业库 ...
随机推荐
- PySpark初级教程——第一步大数据分析(附代码实现)
概述 数据正以前所未有的速度与日俱增 如何存储.处理和使用这些数据来进行机器学习?spark正可以应对这些问题 了解Spark是什么,它是如何工作的,以及涉及的不同组件是什么 简介 我们正在以前所未有 ...
- 《java编程思想》对象导论
1.抽象过程 所有编程语言都提供抽象机制.可以认为,人们所能够解决的问题的复杂性直接取决于抽象的类型和质量,所谓的'类型'是指“所抽象的是什么?”汇编语言是对底层机器的轻微抽象. java的基本 特性 ...
- 关于 JavaScript 的 精度丢失 与 近似舍入
一.背景 最近做 dashborad 图表时,涉及计算小数且四舍五入精确到 N 位.后发现 js 算出来的结果跟我预想的不一样,看来这里面并不简单-- 二.JS 与 精度 1.精度处理 首先明确两点: ...
- SQL实战(二)
一. 获取所有员工当前的manager,如果当前的manager是自己的话结果不显示,当前表示to_date='9999-01-01'.结果第一列给出当前员工的emp_no,第二列给出其manager ...
- 看完这篇Exception 和 Error,和面试官扯皮就没问题了
在 Java 中的基本理念是 结构不佳的代码不能运行,发现错误的理想时期是在编译期间,因为你不用运行程序,只是凭借着对 Java 基本理念的理解就能发现问题.但是编译期并不能找出所有的问题,有一些 N ...
- 性能测试环境搭建:XAMPP1.8+PHPwind9.0安装教程
1.安装准备 1.1软件版本 XAMPP的版本:XAMPP 1.8.2 phpwind的版本:PHPWind 9.0 1.2.安装环境 我的环境:win10 其实安装环境影响不大,win7,win ...
- Vertica的这些事(五)——-谈谈vertica的flex-table
Json格式对于现在所有的软件开发者都不陌生,很多数据格式都用他来存储,我们来看一下vertica是怎么处理json数据的.这就是vertica的flex table! 首先创建一个json文件: { ...
- nexus Maven私服的相关配置
Maven私服中如需本地上传Maven私服内容则需在 setting.xml中配置如下: <server> <id>nexus-releases</id> < ...
- 1066 Root of AVL Tree (25分)(AVL树的实现)
An AVL tree is a self-balancing binary search tree. In an AVL tree, the heights of the two child sub ...
- Y分形的平面微带天线生成过程
Y分形的平面微带天线生成过程 本文介绍了使用Altium Designer脚本程序生成Y型天线的过程,在窗体中线宽.迭代次数,边框长度可以直接设置. Y分形天线用户界面由一个窗体.1个TImage控件 ...