1 PointCut 由 ClassFilter 和 MethodMatcher 构成,通过 ClassFilter 定位到类上,通过 MethodMatcher 定位到方法。

2 Spring 支持两种方法匹配器:静态方法匹配器和动态方法匹配器。

静态方法匹配器仅对方法名签名进行匹配;动态方法匹配器会在运行期检查方法入参的值。

静态匹配仅会匹配一次;动态匹配因为每次调用方法的入参可能不一样,所以每次调用都需要判断。因此动态匹配对性能影响比较大。

3 Advisor 既包含切点代码又包含连接点信息(方法前、方法后)。切面可以分为三类:一般切面、切点切面和引介切面

  • Advisor:代表一般切面,仅包含一个 Advice ,因为 Advice 包含了横切代码和连接点信息。他代表的连接点是所有目标类的所有方法,因为该横切面太宽泛,所以一般不用。
  • PointcutAdvisor:代表具有切点的切面,包含 Advice 和 Pointcut 两类。
  • IntroductionAdvisor:引介切面,它应用于类层面上,所以使用 ClassFilter 定义

4 基于配置使用@AspectJ 切面

在配置文件中使用

<aop:aspectj-autoproxy />

自动为 Spring 容器中那些匹配 @AspectJ 切面的 Bean 创建代理,完成切面织入。在 Spring 内部使用 AnnotationAwareAspectJAutoProxyCreator 进行自动代理的创建。

它有一个属性 proxy-target-class,默认是 false,表示使用jdk 动态代理。当设置为 true 时,表示使用 CGLib 动态代理技术。不过即使设置为 false,如果目标类中没有接口,Spring 自动使用

CGLib 动态代理。

<aop:aspectj-autoproxy proxy-target-class="true" />

5 Spring 支持9个 @AspectJ 切点表达式函数,根据描述对象的不同,可以分为 4 种类型

spring 中 AOP 功能-LMLPHP

spring 中 AOP 功能-LMLPHP

spring 中 AOP 功能-LMLPHP

在多个表达式之间使用  || , or 表示  或 ,使用  && , and 表示  与 , ! 表示 非

例如:

@Pointcut("execution(* com.huitong.controller..*.* (..)) " +
"&& @annotation(org.springframework.web.bind.annotation.RequestMapping)")
public void controllerPoint() {
}

6 切点函数入参使用通配符, @AspectJ 支持3种通配符。

  • “*”,匹配任意字符,但它只能匹配上下文中一个元素。
  • “..”,匹配任意字符,可以匹配多个元素;但在表示类时,必须和 * 联合使用。
  • “+”,按类型匹配指定类的所有类,必须跟在类名后面,如 com.smart.Car+。继承或扩展指定类的所有类,同时还包括指定类的本身。

支持所有通配符:execution(),within()

仅支持“+”通配符:args(),this(),target()

不支持通配符:@args(),@within(),@target()

7 @AspectJ 提供的增强类型

  • @Before ,该注解拥有两个成员,1)value用于定义切点  2)argNames,注解所标注增强方法的参数名
  • @AfterReturning 后置增强,有4个成员。1)value 用于定义切点  2)pointcut 表示切点信息。如果显示指定 pointcut 值,那么他将覆盖 value 的设置值。可将 pointcut 看做 value 同义词  3)returning 将目标对象方法的返回值绑定给增强的方法  4)argNames:如前所述
  • @Around,环绕增强,两个成员 1)value 用于定义切点  2)argNames:如前所述
  • @AfterThrowing,抛出增强,1)value 用于定义切点 2)pointcut 表示切点信息,如果显示指定 pointcut 值,那么他将覆盖 value 的设置值。  3)throwing 将抛出的异常绑定到增强方法中 4)argNames:如前所述
  • @After:不管是抛出异常还是正常退出,该增强都会得到执行。有两个成员,1)value 用于定位切点  2)argNames:如前所述

8 execution()切点函数,需要定位到方法

“.*”:表示包下所有类,“..*”:表示包、子孙包下的所有类。

within(),类匹配模式串,只能定位到类,且不是接口。

9 命名切点:使用 @Pointcut 注解及切面类方法对切点进行命名。

如果在增强类型里面直接定义切点,这种称为匿名切点。

例子如下:

spring 中 AOP 功能-LMLPHP

命名切点是用类方法作为切点名称,方法的访问修饰符控制了切点的可引用性。

spring 中 AOP 功能-LMLPHP

10 定义好切点后就可以在切面中通过名称引用切点。例子如下

spring 中 AOP 功能-LMLPHP

11 访问连接点信息 AspectJ 使用 JoinPoint 接口表示目标类连接点对象。如果是环绕增强,则使用 ProceedingJoinPoint 表示连接点对象,该类是

JoinPoint 子接口。任何增强方法都可以通过第一个参数声明为 JoinPoint 访问连接上下文信息。

spring 中 AOP 功能-LMLPHP

11.2 ProceedingJoinPoint 继承于 JoinPoint 接口,它新增两个用于执行连接点的方法。

spring 中 AOP 功能-LMLPHP

例子如下:

spring 中 AOP 功能-LMLPHP

05-11 08:21