【1】Aspect优先级
在同一个连接点上应用不止一个切面时, 除非明确指定, 否则它们的优先级是不确定的。
切面的优先级可以通过实现 Ordered
接口或利用 @Order
注解指定.
实现 Ordered 接口, getOrder() 方法的返回值越小, 优先级越高; 若使用 @Order 注解, 序号出现在注解中。
【2】@Pointcut重用切入点定义
在编写 AspectJ 切面时, 可以直接在通知注解中书写切入点表达式。但同一个切点表达式可能会在多个通知中重复出现。
在 AspectJ 切面中, 可以通过 @Pointcut 注解将一个切入点声明成简单的方法。切入点的方法体通常是空的, 因为将切入点定义与应用程序逻辑混在一起是不合理的。
切入点方法的访问控制符同时也控制着这个切入点的可见性。
如果切入点要在多个切面中共用, 最好将它们集中在一个公共的类中。在这种情况下, 它们必须被声明为 public。 在引入这个切入点时, 必须将类名也包括在内。 如果类没有与这个切面放在同一个包中, 还必须包含包名。
其他通知可以通过方法名称引入该切入点。匹配ArithmeticCalculatorImpl下参数为(int,int)的所有方法:
@Pointcut("execution(int com.web.aop.impl.ArithmeticCalculatorImpl.*(int , int ))") public void declareJoinPointExpression() { }
匹配controller包下的所有类的所有方法:
@Pointcut("execution(* com.recommend.controller.*.*(..))") public void declareJoinPointExpression() { }
【3】AspectJ 切入点表达式
最典型的切入点表达式时根据方法的签名来匹配各种方法
execution * com.web.spring.ArithmeticCalculator.*(..)
匹配 ArithmeticCalculator 中声明的所有方法; 第一个 * 代表任意修饰符及任意返回值. 第二个 * 代表任意方法; '..' 匹配任意数量的参数. 若目标类与接口与该切面在同一个包中, 可以省略包名.
execution public * ArithmeticCalculator.*(..)
匹配 ArithmeticCalculator 接口的所有公有方法.
execution public double ArithmeticCalculator.*(..)
匹配 ArithmeticCalculator 中返回 double 类型数值的方法
execution public double ArithmeticCalculator.*(double, ..)
匹配第一个参数为 double 类型,结果返回类型为double类型的方法, ‘..’ 匹配任意数量任意类型的参数
execution public double ArithmeticCalculator.*(double, double)
匹配参数类型为 double, 返回double 类型的方法.
【4】其他的切入点类型完整的AspectJ切入点语言支持额外的切入点指定者,但是Spring不支持这个功能。 他们分别是call, initialization, preinitialization, staticinitialization, get, set, handler, adviceexecution, withincode, cflow, cflowbelow, if, @this 和 @withincode。
在Spring AOP中使用这些指定者将会导致抛出IllegalArgumentException异常。
Spring AOP支持的切入点指定者可能在将来的版本中得到扩展,不但支持更多的AspectJ 切入点指定者(例如"if"),还会支持某些Spring特有的切入点指定者,比如"bean"(用于匹配bean的名字)。
execution - 匹配方法执行的连接点,这是你将会用到的Spring的最主要的切入点指定者。
within - 限定匹配特定类型的连接点(在使用Spring AOP的时候,在匹配的类型中定义的方法的执行)。
this - 限定匹配特定的连接点(使用Spring AOP的时候方法的执行),其中bean reference(Spring AOP 代理)是指定类型的实例。
target - 限定匹配特定的连接点(使用Spring AOP的时候方法的执行),其中目标对象(被代理的appolication object)是指定类型的实例。
args - 限定匹配特定的连接点(使用Spring AOP的时候方法的执行),其中参数是指定类型的实例。
@target - 限定匹配特定的连接点(使用Spring AOP的时候方法的执行),其中执行的对象的类已经有指定类型的注解。
@args - 限定匹配特定的连接点(使用Spring AOP的时候方法的执行),其中实际传入参数的运行时类型有指定类型的注解。
@within - 限定匹配特定的连接点,其中连接点所在类型已指定注解(在使用Spring AOP的时候,所执行的方法所在类型已指定注解)。
@annotation - 限定匹配特定的连接点(使用Spring AOP的时候方法的执行),其中连接点的主题有某种给定的注解。
因为Spring AOP限制了连接点必须是方法执行级别的,pointcut designators的讨论也给出了一个定义,这个定义和AspectJ的编程指南中的定义相比显得更加狭窄。
除此之外,AspectJ它本身有基于类型的语义,在执行的连接点’this’和’target’都是指同一个对象,也就是执行方法的对象。
Spring AOP是一个基于代理的系统,并且严格区分代理对象本身(对应于’this’)和背后的目标对象(对应于’target’)
【5】合并切入点表达式
切入点表达式可以使用'&&', '||' 和 '!'
来合并。还可以通过名字来指向切入点表达式。
以下的例子展示了三种切入点表达式:
anyPublicOperation(在一个方法执行连接点代表了任意public方法的执行时匹配); inTrading(在一个代表了在交易模块中的任意的方法执行时匹配) ; tradingOperation(在一个代表了在交易模块中的任意的公共方法执行时匹配)。
实例如下:
@Pointcut("execution(public * *(..))") private void anyPublicOperation() {} @Pointcut("within(com.someapp.trading..*") private void inTrading() {} @Pointcut("anyPublicOperation() && inTrading()") private void tradingOperation() {}
就上所示的,从更小的命名组件来构建更加复杂的切入点表达式是一种最佳实践。 当用名字来指定切入点时使用的是常见的Java成员可视性访问规则。
比如说,你可以在同一类型中访问私有的切入点,在继承关系中访问受保护的切入点,可以在任意地方访问公共切入点。 成员可视性访问规则不影响到切入点的 匹配。