这两天忙着看AspectJ in Action 为了补一下AOP知识。看了Spring 2.0的规范,其中AOP部分已经基本融合了AspectJ,看来有必要看一看AspectJ了。
看了很多AOP的文章了,AOP这两年发展的很慢,没有什么新意,现在到处都是SOA,SCA了,不过研究了一下,觉得还是很有帮助的。尤其是增加系统的契约性和模块的独立性来说,很有帮助。
当然,学东西,基础很重要。下面就说说AspectJ中的基本语法,有兴趣的可以看看AspectJ in Action。
先来说说pointcut,从字面的意思说的是切面的意思。也就是横切的时候,会有哪些执行点会被识别。只有先识别了,才能执行相应的Advice。
基本的定义如下:
public
pointcut accountOperations:call(* Account.*(..))
1.通配符和pointcut 操作符
- * 表示任何数量的字符,除了(.)
- .. 表示任何数量的字符包括任何数量的(.)
- + 描述指定类型的任何子类或者子接口
同java一样,提供了一元和二元的条件表达操作符。
一元操作符:!
二元操作符:||和&&
优先权同java
2.签名语法
类型签名样式
主要的例子: Account 类型Account
*Account 使用Account名称结束的类型,如SavingsAccount和CheckingAccount
java.*.Date 类型Date在任何直接的java子包中,如java.util.Date和java.sql.Date
java..* 任何在java包或者所有子包中的类型,如java.awt和java.util或者java.awt.event 和java.util.logging
javax..*Model+ 所有javax包或者子包中以Model结尾的类型和其所有子类,如TableModel,TreeModel。
!vector 所有除了Vector的类型
Vector|| Hashtable Vector或者Hashtable类型
java.util.RandomAccess+ 实现RandomAccess和List的所有子类 && java.util.List+
方法和构造器签名模式
public void Collection.clear():
在Collection中同样签名的clear方法
public void Account.debit(float) throws InsufficientBalanceException:
Account中同样签名的debit方法
public void Account.set*(*)
Account中以set开头,并且只有一个参数类型的方法
public void Account.*()
Account中所有的没有参数的public void 方法
public * Account.*()
Account中所有没有参数的public 方法
public * Account.*(..)
Account中所有的public 方法
* Account.*(..)
Account中的所有方法,包括private方法
!public * Account.*(..)
所有的非public 方法
* Account+.*(..)
所有的方法,包括子类的方法
* java.io.Reader.read(..)
所有的read方法
* java.io.Reader.read(char[],..)
所有以read(char[])开始的方法,包括read(char[])和read(char[],int,int)
* javax..*.add*Listener(EventListener+)
命名以add开始,以Listener结尾的方法,参数中为EventListener或子类
* *.*(..) throws RemoteException
抛出RemoteException的所有方法
构造器,同上面
public Account.new()
没有参数的构造器方法
属性签名模式
同方法一样,属性也查不多
* Account.*
所有的Account属性
!public static * banking..*.*
所有的非public static 属性,在banking的包或者子包中
3.主要的pointcuts类型
分类pointcuts
遵循特定的语法用于捕获每一个种类的可使用连接点。
主要的种类:
- 方法执行:execution(MethodSignature)
- 方法调用:call(MethodSignature)
- 构造器执行:execution(ConstructorSignature)
- 构造器调用:call(ConstructorSignature)
- 类初始化:staticinitialization(TypeSignature)
- 属性读操作:get(FieldSignature)
- 属性写操作:set(FieldSignature)
- 例外处理执行:handler(TypeSignature)
- 对象初始化:initialization(ConstructorSignature)
- 对象预先初始化:preinitialization(ConstructorSignature)
- Advice执行:adviceexecution()
基于控制流的pointcuts 主要包括两种类型的控制流:
cflow(Pointcut),捕获所有的连接点在指定的方法执行中,包括执行方法本身。
cflowbelow(Pointcut),捕获所有的连接点在指定的方法执行中,除了执行方法本身。
如以下的例子:
cflow(call(* Account.debit(..)) 所有的debit方法中的连接点,包括debit方法本身
cflowbelow(call(* Account.debit(..)) 所有debit方法中的连接点,除了debit方法本身
cflow(transactedOperations()) 所有由transactedOperations捕获的连接点
cflowbelow(execution(Account.new(..)) 所有在Account 构造器中执行的连接点
基于词汇结构的连接点 源代码片断。,如within()和withincode()
within :捕获在指定类或者方面中的程序体中的所有连接点,包括内部类。
Withincode:用于捕获在构造器或者方法中的所有连接点,包括在其中的本地类
执行对象连接点 匹配this,和target对象,作为方法被调用的对象。
this(Account),所有Account的实例的执行点,匹配所有的连接点,如方法调用,属性设置,当前的执行对象为Account,或者其子类。
target(Account):匹配所有的连接点,目标对象为Account或其子类。
必须执行相应的类型,不能使用*,或者..通配符。当前静态方法,不能被匹配。
在within()和this()中的区别:
一个是程序体,而另一个为对象执行。
参数pointcuts 用于捕获参数类型的连接点。
args(String,..,int)
args(RemoteException)
条件检测pointcuts if(System.currentTimeMillis()>triggerTime)
基本的语法就这些,不过AspectJ5 已经增加了不少语法,主要关于Annotation 的,不过这部分还没有仔细研究,以后再讲解吧。希望能有所帮助