AspectJ 是为了解决面向切面的编程而诞生的,通过它可以在Java应用程序中织入横切关注点,比如日志、性能分析、安全检查等。AspectJ 定义了一套自己的语法,拥有自己的编译器。
AOP与AspectJ以及SpringAOP的关系
- AOP: Aspect-Oriented Programming 即面向切面的编程,是一种编程模式,或者说是一种编程思想.
- AspectJ:是一个实现了 AOP 的框架,它提供了一种基于 Java 语言的面向切面编程语言扩展。AspectJ 具有强大的 AOP 功能,可以在编译期、类加载期或运行时织入切面,从而使开发人员能够更加灵活地控制切面的应用时机和范围
- Spring AOP 是 Spring 框架中的 AOP 模块,是对Spring 管理的 Bean 进行切面编程的一种实现,并且完全支持 AspectJ 的语法,除此之外,与 AspectJ 没有关系,它是通过动态代理(JDK Proxy 或者 CGLib)来实现的.
它们三者的关系如下图所示:
AspectJ 代码织入的方式
AspectJ 代码织入的方式有三种:
-
编译时织入(Compile-time weaving, CTW): 在编译目标程序时,将切面代码编译到目标代码中,这种方式需要使用特殊的编译器来支持,比如 Aspect 编译器:ajc.
-
类装载时织入(Load-time weaving, LTW): 当目标程序加载到 JVM 时,通过特殊的代理机制动态织入切面代码。这种方式需要使用特殊的代理库来支持,例如 AspectJ Weaver(aspectjweaver.jar). 一般都是通过 javaagent 来实现。
-
运行时织入(Runtime weaving, RTW): 在目标程序运行时,通过特殊的API动态织入切面代码,这种方式需要使用 AspectJ 提供的 API 来实现,例如 AspectJProxyFactory 类. 其中,编译时织入和类装载时织入,需要在编译和运行时使用特殊的参数和配置来启用,而运行时织入可以通过代码动态实现.
一般来说,编译时织入和类装载时织入可以在编译时或打包时完成,从而可以提高目标程序的性能。而运行时织入则可以更灵活地动态地控制切面的织入和撤销,但可能会影响目标程序的性能。
AspectJ 语法
AspectJ 语法分为三种
- 直接使用 AspectJ 语法编写 aspect 类
- 基于 XML 配置文件的语法
- 注解的语法
编写aspect类
这种方式的切面类,通常以 .aj 结尾,并且更加灵活, 例如,以下是一个使用 .aj 文件编写的切面类的示例:
public aspect MyAspect {
pointcut myPointcut(): execution(* com.example.MyClass.myMethod(..));
before(): myPointcut() {
// advice code
}
}
注意,在使用 .aj 文件编写切面时,需要使用特定的编译器(例如 ajc 命令)进行编译,而不能直接使用普通的 Java 编译器(例如 javac 命令)。另外,由于 .aj 文件使用的是 AspectJ 语法而不是标准的 Java 语法,因此需要在编译时指定 AspectJ 运行时库的类路径,例如:
ajc -cp /path/to/aspectjrt.jar MyAspect.aj
这个命令将编译 MyAspect.aj 文件,并将 /path/to/aspectjrt.jar 文件添加到类路径中,以便编译器可以查找 AspectJ 运行时库中的类和方法。
实战
- 安装 ajc 编译器
我们首先需要下载 AspectJ 的编译器 ajc, 可以通过 eclipse 官网进行下载:
https://www.eclipse.org/aspectj/downloads.php
下载完成后,运行下面的命令进行安装:
java -jar aspectj-1.x.x.jar
安装完成后,还需添加环境变量,之后在命令行运行 ajc,check是否安装成功.
- 编写要被织入的Java目标代码
public class UserService {
public int addUser() {
System.out.println("add user");
return 1;
}
}
public class AjcTest {
public static void main(String[] args) {
UserService userService = new UserService();
userService.addUser();
}
}
- 编写切面类
public aspect AuthAspect {
before():execution(* com.kklab.clouddemo.ajc.*.*(..)) {
System.out.println("process auth check");
}
}
- 编译
ajc -cp /path/to/aspectjrt.jar -d *.java *.aj
-d 命令可以让class文件在当前文件夹下生成
生成的UserService.class文件如下, 可以看到在 addUser 方法中已经织入了一行代码,这行代码将会调用我们定义的切面方法.
public class UserService {
public UserService() {
}
public int addUser() {
AuthAspect.aspectOf().ajc$before$com_kklab_clouddemo_ajc_AuthAspect$1$204a3380();
System.out.println("add user");
return 1;
}
}
假如我们将 aspect 类中 execution 改为 call,再运行生成 class 文件会发现,并没有在 UserService.class 中织入代码,而是在调用 UserSerivce 方法的地方织入了代码:
public class AjcTest {
public AjcTest() {
}
public static void main(String[] args) {
UserService userService = new UserService();
AuthAspect.aspectOf().ajc$before$com_kklab_clouddemo_ajc_AuthAspect$1$1dafce7a();
userService.addUser();
}
}
这也充分说明了 execution 和 call 两者的区别:
execution 用在方法执行的位置织入,而 call 是在方法调用的位置织入.
在编写切面时,需要根据具体的需求选择合适的切入点类型。
注解语法
AspectJ 的注解语法是基于 Java 注解的,可以使用 @Aspect 注解声明一个切面类,使用其他注解声明切点和通知等元素。 以下是常用的 AspectJ 注解及其用法:
- @Aspect:用于声明一个切面类。
- @Pointcut:用于声明一个切点
- @Before:用于声明一个前置通知。
- @After:用于声明一个后置通知。
- @Around:用于声明一个环绕通知。
- @AfterReturning:用于声明一个返回通知。
- @AfterThrowing:用于声明一个异常通知。
除了以上注解外,AspectJ 还提供了其他一些注解,例如 @DeclareParents 用于声明一个引入通知,以及 @DeclarePrecedence 用于声明切面类之间的优先级顺序等。
实战
我们创建一个普通的Java类文件,不过这次要在这个类上面加一些 aspect 的注解:
@Aspect
public class LogAspect {
@Before("execution(* com.kklab.clouddemo.ajc.*.*(..))")
public void logBefore() {
System.out.println("Before executing add method.");
}
}
然后再使用 ajc 编译一下,由于注解是在jdk 1.5 之后才出现的,所以这次我们需要指定一下 ajc 的 jdk 兼容模式:
ajc -cp /path/to/aspectjrt.jar -d *.java *.aj -1.8
编译之后,查看 class 文件,发现同样对 UserService.class 进行了织入:
public class UserService {
public UserService() {
}
public int addUser() {
AuthAspect.aspectOf().ajc$before$com_kklab_clouddemo_ajc_AuthAspect$1$204a3380();
LogAspect.aspectOf().logBeforeAdd();
System.out.println("add user");
return 1;
}
}
AspectJ 表达式语法
AspectJ 表达式语法是用来声明切入点的,可以指定匹配的目标方法或者构造方法。 AspectJ 表达式语法非常的灵活,可以根据需要使用各种模式匹配规则来定位目标 JoinPoint, 下面是 AspectJ 表达式语法的一些常用模式:
方法调用模式
call([可见性模式] [返回类型模式] 包名模式.类名模式.方法名模式([参数模式]) [throws模式])
示例:
// 匹配可见性为 public,返回类型为 void,包名为 com.example,类名以 Service 结尾,方法名以 save 开头,且有一个参数的方法。
call(public void com.example.*Service.save(*))
// 匹配所有可见性,返回类型为 int,包名为 com.example,类名以 Dao 结尾,方法名为 get 开头,且不接受参数的方法。
call(int com.example.Dao.get())
// 匹配可见性为 private 或 protected,返回类型为任意类型,包名为 com.example,类名以 *Dao 结尾,方法名为 *,且任意参数的方法。
call(private|protected * com.example.*Dao.*(*))
方法执行模式
execution([可见性模式] [返回类型模式] 包名模式.类名模式.方法名模式([参数模式]) [throws模式])
示例:
// 匹配可见性为 public,返回类型为 void,包名为 com.example,类名以 Service 结尾,方法名以 save 开头,且有一个参数的方法。
execution(public void com.example.*Service.save(*))
// 匹配所有可见性,返回类型为 int,包名为 com.example,类名以 Dao 结尾,方法名为 get 开头,且不接受参数的方法。
execution(int com.example.Dao.get())
// 匹配可见性为 private 或 protected,返回类型为任意类型,包名为 com.example,类名以 *Dao 结尾,方法名为 *,且任意参数的方法。
execution(private|protected * com.example.*Dao.*(*))
Maven 自动化构建
codehaus 提供了一个 ajc 的编译插件 aspectj-maven-plugin,只要在 maven 配置文件中添加这个插件, 就可以使用 maven 来编译 AspectJ 了.
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>aspectj-maven-plugin</artifactId>
<version>1.10</version>
<configuration>
<source>1.8</source>
<target>1.8</target>
<complianceLevel>1.8</complianceLevel>
</configuration>
<executions>
<execution>
<goals>
<goal>compile</goal>
</goals>
</execution>
</executions>
</plugin>
参考文档: https://toutiao.io/posts/os3asg/preview