本文适用于所有AspectJ与Spring AOP混用的场景。
Spring AOP 是基于动态代理的实现AOP,基于 JDK代理和CGLib代理实现运行时织入(runtime weaving)。
Spring AOP的切面定义沿用了ASpectJ的注解体系,所以在Spring体系中注解定义切面的方式与ASpectJ完全兼容, 但如果一个项目中有Spring 切面定义又要使用aocache静态态织入(CTW)的情况下,就会存在冲突问题。
如下面的示例,项目中定义了一个Spring AOP切面,拦截所有带@RestController
注解的服务方法
import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.springframework.stereotype.Component; /** * * 服务上下文切面 * */ @Aspect @Component public class ServiceContextAop { @Before("restControllerAspect()") public void doServiceContextBefore(JoinPoint joinPoint) throws Throwable { // DO SOMETHING } /** * 定义切入点为有注解{@link org.springframework.web.bind.annotation.RestController}下的所有类 */ @Pointcut("@within(org.springframework.web.bind.annotation.RestController)") public void restControllerAspect() { } }
pom.xml中如下定义了aspectj-maven-plugin
插件在项目编译时静态织入aocache定义的切面
org.codehaus.mojo aspectj-maven-plugin 1.10 1.8 UTF-8 1.8 true true com.gitee.l0km aocache compile
编译时不会有任何问题,但是在程序运行时,就会发现ServiceContextAop
定义的切面没有起作用。
因为aspectj-maven-plugin
插件识别到了ServiceContextAop
类上的@Aspect
注解,将本该由Spring AOP管理的切面也执行了静态织入。如下为ServiceContextAop.class
反编译显示的代码,可以看到ASpectJ在类中增加了额外代码。
为解决ASpectJ CTW模式与Spring AOP的共存问题,就要在aspectj-maven-plugin
插件中使用
参数指定在执行静态织入时排除ServiceContextAop.java
org.codehaus.mojo aspectj-maven-plugin 1.10 1.8 UTF-8 1.8 true true com.gitee.l0km aocache **/ServiceContextAop.java compile
关于
参数说明参见
《Using includes/excludes》