Castle实践8-AspectSharp (转自叶子)
AOP是基于动态代理技术的。在学习AOP之前必须明白几个概念,这几个概念我将由配置AspectSharp(以下简称A#)的配置文件中一一引出,包括:Advice、Pointcut、Advister、Mixin 。
A#有自己独特的配置语言,当然也支持XML,但是新的配置语言我觉得比XML更加明了直观,而且也是非常容易使用的。 详细的官方文档在这里:http://www.castleproject.org/index.php/AspectSharp_Language_Documentation
1)配置必须按照以下顺序:
[Global Interceptor map]
[Global Mixin map]
Aspects definitions
2)[Imports]:引入命名空间,在下面的配置中用到的拦截器、混淆器所需要的。
3)[Global Interceptor map]:如果你想在程序共享同一个拦截器而不想重复声明可以将Interceptor声明为全局,在同一配置文件中重用,而不用再次打长长的名称,用声明的别名就可以了。
"key" : InterceptorType ;
"key2" : InterceptorType2
]
4)[Global Mixin map] : 同样混淆器也可以声明为全局。
"key" : MixinType ;
"key2" : MixinType2
]
5)Aspects definitions : 具体定义一个“切面(需要拦截的地方)”
[include]
[pointcuts]
end
6)[include] :定义混淆器(mixin)组合的类
include DigitalGravity.Mixins.Security in DigitalGravity.XProject
include System.Collections.ArrayList in System
end
7)[pointcuts] :拦截的具体名称,这里先指定拦截的类型并可以用通配符匹配名称。类型如下:
end
pointcut method|propertyread(*)
end
pointcut propertywrite(*)
end
8)Advices :指定由哪个拦截器拦截
pointcut method(*)
advice(DigitalGravity.Interceptors.LogInvocationInterceptor in DigitalGravity.XProject)
end
end
上面基本是官方的配置例子,那么这几个咚咚有什么关系,引用一句话“advice是你想向别的程序内部不同的地方注入的代码。pointcut定义了需要注入advice的位置,通常是某个特定的类的一个public方法。advisor是pointcut和advice的装配器,是将advice注入主程序中预定义位置的代码。”这句话是从java关于spring文章中截取出来的,原文在这里:http://www.wjshome.com/bbs/topic.aspx?topicid=209&page=65535,上面所讲的advice和pointcut的概念和这里是一样的,对于advisor我想是相对于上面讲的Aspects definitions 。这样大家就非常容易理解了吧。
详细了解动态代理请看:http://iaxes.cnblogs.com/archive/2005/04/07/132868.html
详细了解Maxin请看:http://iaxes.cnblogs.com/archive/2005/04/07/133407.html
AOP基本资料请看:http://www.wjshome.com/bbs/topic.aspx?topicid=203
AOP-面向方面编程,定义这里我不想介绍太多,如果有兴趣的话,你可以来:http://www.wjshome.com/bbs/board.aspx?boardid=14看看aop的资料。那么AOP主要应用在哪些方面呢?具体来说,有:
Authentication--权限、Caching--缓存、Context passing--内容传递、Error handling--错误处理、Lazy loading--延迟加载、Debugging--调试、logging tracing profiling and monitoring--记录跟踪 优化 校准、Performance optimization--性能优化、Persistence--持久化、Resource pooling--资源池、Synchronization--同步、Transactions--事务……。
A#是CastleProject在.Net上实现AOP的一个框架,另外Spring.Net也有AOP的咚咚。 其实在Java里面,aop已经是很久以前的话题了,只是。net方面资料不如java多,用的人也相对少了。我下面将以一个简单的例子(参照官方改写),阐述A#在.net下面实现简单的“日志”和“权限检查”的AOP编程。
1)配置文件:
import AopDemo.Mixins
aspect log for [AopDemo]
pointcut method(*)
advice(LoggerInterceptor)
end
end
aspect Security for [AopDemo]
include SecurityMixin
pointcut method(*)
advice(SecurityCheckInterceptor)
end
end
最上面两个import是引入命名空间,是LoggerInterceptor、SecurityCheckInterceptor和SecurityMixin的。我发现如果在同一个程序集中,不使用import也行,但是如果你的interceptor是单独的dll的话,应该必须import,我没有尝试。把interceptor独立出来的好处就是可以很方便的替换他。
接着是两个“切面”定义,第一个是定义所有的方法都会被LoggerInterceptor拦截,用于记录日志。上面讲到可以利用通配符来筛选拦截的方法名称,所以“*”就是代表所有的方法,又比如说:pointcut void Create(*)就是拦截返回类型为void,参数不限制的Create方法,void create(int id), void create(string name)这两个方法都会被拦截。第二个pointcut是加入了一个权限混淆器SecurityMixin,他的所有方法调用会被SecurityCheckInterceptor拦截。SecurityMixin的代码如下:
{
private object proxy;
public SecurityMixin()
{}
#region ISecurity 成员
public bool Access
{
get
{
// use proxy to do sth.
// eg: if (proxy is PersonDao)
return true;
}
}
#endregion
#region IProxyAware 成员
public void SetProxy(object proxy)
{
this.proxy = proxy;
}
#endregion
}
他实现了两个接口:ISecurity和IProxyAware。
ISecurity有一个Access属性,而实现IProxyAware是为了在mixin中使用代理对象。这样定义之后,在SecurityCheckInterceptor拦截的代理对象就是实现了ISecurity接口的对象,可以调用ISecurity接口中定义的方法了,而原来的对象是没有实现这个接口的。
使用的时候先加载配置初始化AspectEngine:
AspectEngine engine = builder.Build();
然后通过这样来使用:
dao.Create("CCC");
Create方法将被拦截咯,作日志,安全检查,随便你~
好了,就说到这吧,下载例子来调试调试咯:
http://www.wjshome.com/Income/Others/Castle.AopDemo.rar
注意例子中的PersonStore和PersonDao是模拟数据库的操作的,方便大家调试。bye~