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

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

流烟默

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值