0
点赞
收藏
分享

微信扫一扫

SpringAOP中Aspect优先级与切面表达式详解

【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成员可视性访问规则。

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


举报

相关推荐

0 条评论