Spring - Aspect优先级与切面表达式

【1】Aspect优先级

在同一个连接点上应用不止一个切面时, 除非明确指定, 否则它们的优先级是不确定的。

切面的优先级可以通过实现 Ordered 接口或利用 @Order 注解指定.

实现 Ordered 接口, getOrder() 方法的返回值越小, 优先级越高;
若使用 @Order 注解, 序号出现在注解中。

【2】重用切入点定义

在编写 AspectJ 切面时, 可以直接在通知注解中书写切入点表达式. 但同一个切点表达式可能会在多个通知中重复出现.

在 AspectJ 切面中, 可以通过 @Pointcut 注解将一个切入点声明成简单的方法. 切入点的方法体通常是空的, 因为将切入点定义与应用程序逻辑混在一起是不合理的.

切入点方法的访问控制符同时也控制着这个切入点的可见性.。

如果切入点要在多个切面中共用, 最好将它们集中在一个公共的类中. 在这种情况下, 它们必须被声明为 public. 在引入这个切入点时, 必须将类名也包括在内. 如果类没有与这个切面放在同一个包中, 还必须包含包名.

其他通知可以通过方法名称引入该切入点.

@Pointcut("execution(int com.web.aop.impl.ArithmeticCalculatorImpl.*(int , int ))")
public void declareJoinPointExpression() {

}

【3】AspectJ 切入点表达式

最典型的切入点表达式时根据方法的签名来匹配各种方法:

(1).execution * com.atguigu.spring.ArithmeticCalculator.*(..)

匹配 ArithmeticCalculator 中声明的所有方法;
第一个 * 代表任意修饰符及任意返回值. 
第二个 * 代表任意方法;
'..' 匹配任意数量的参数. 
若目标类与接口与该切面在同一个包中, 可以省略包名.

(2).execution public * ArithmeticCalculator.*(..)

匹配 ArithmeticCalculator 接口的所有公有方法.


(3).execution public double ArithmeticCalculator.*(..)

匹配 ArithmeticCalculator 中返回 double 类型数值的方法


(4).execution public double ArithmeticCalculator.*(double, ..)


匹配第一个参数为 double 类型,结果返回类型为double类型的方法, ‘..’ 匹配任意数量任意类型的参数

(5).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.xyz.someapp.trading..*")
    private void inTrading() {}



    @Pointcut("anyPublicOperation() && inTrading()")
    private void tradingOperation() {}

就上所示的,从更小的命名组件来构建更加复杂的切入点表达式是一种最佳实践。 当用名字来指定切入点时使用的是常见的Java成员可视性访问规则。

比如说,你可以在同一类型中访问私有的切入点,在继承关系中访问受保护的切入点,可以在任意地方访问公共切入点。 成员可视性访问规则不影响到切入点的 匹配。

你可能感兴趣的:(Spring)