Spring AOP

目录

AOP

理解AOP

AOP组成

AOP的优点

Spring AOP

使用Spring AOP

定义切面和切点

定义通知

动态代理 

织入


AOP

理解AOP

AOP即面向切面编程,简单来说,就是把一部分通用的功能集中的放在一个地方处理的思想。假如某一段代码很多地方要用到,比如说登录验证,在传统编程中,有两个做法,要么每次复制粘贴,要么把他封装成一个函数再调用。复制粘贴显然是最坏的一种做法,一旦涉及到修改就会很麻烦。因此封装成函数是一种更优的做法。而AOP可以看作是一种更为高级和抽象的封装,它可以动态地植入通用功能。Spring AOP是AOP思想的一种具体实现。

AOP组成

AOP的基本组成包含切面(Aspect)、切点(Pointcut)、连接点(Joinpoint)、通知(Advice)。

  • 切面(Aspect):由连接点、切点和通知组成。简单来说切面就是一个包含了连接带你、切点和通知的类。即实现某个或某些功能的集合。
  • 连接点(Joinpoint):程序运行时可以切入切面执行通知的点。例如执行方法前、执行方法后、程序异常时等都可以作为连接点。
  • 切点(Pointcut):提供规则,匹配符合要求的连接点。满足切点规则的连接点才可以切入切面执行逻辑。
  • 通知(Advice):包括切入切面后执行的功能和执行时机。可以使用注解的方式来确定执行时机。

AOP的优点

  • 提高代码复用性和可维护性: AOP可以像函数封装那样,不必重复写相同的代码;
  • 提高开发的效率:AOP可以把不同的业务功能分块,例如日志、安全等模块分成不同的切面,在每个切面处理特定的功能;

Spring AOP

Spring AOP虽然属于Spring全家桶的产品,但是在创建项目时并不能找到相应的依赖:

Spring AOP_第1张图片

 因此需要自行添加(注意版本号):



    org.springframework.boot
    spring-boot-starter-aop
    2.7.13

使用Spring AOP

定义切面和切点

使用@Aspect注解表示当前类为一个切面,使用@Pointcut注解定义切点(即设定规则):

@Component
@Aspect
public class ExperimentalAspect {

    @Pointcut("execution(* com.example.demo.controller.ExperimentalController.*(..))")
    public void pointcut() {}// 空方法,只起到标识的作用

}

@Pointcut注解的参数是一个切点表达式,用于匹配连接点。使用AspectJ语法。切点表达式由切点函数组成,如:execution()、within()、target()、@annotation()等,其中最常用的时execution()函数,用来匹配方法连接点,语法为:

execution(<访问权限修饰符(可省略)><返回类型><包.类.方法(参数)><异常(可省略)>)

访问权限修饰符省略不写表示匹配所有权限,除参数部分外任意一个部分都可以使用通配符'*'来匹配任意字符,例如返回类型使用'*'表示任意返回值都满足规则。

参数中写具体的类型例如(int)表示参数为一个整型,使用'..'表示任意参数都满足。

上文的@Pointcut参数的含义就是匹配任意权限且在ExperimentalController包中的任意方法(返回值任意,参数任意)

其他切点函数用法可以参考Spring官网:Declaring a Pointcut :: Spring Framework

定义通知

Spring AOP通知注解包括:

  • @Before——方法执行前执行通知
  • @After——方法执行后执行通知(方法执行失败也会执行通知)
  • @AfterReturning——方法返回后(成功执行方法)执行通知
  • @AfterThrowing——抛出异常时执行通知
  • @Around——方法执行前后执行通知
// 执行前置通知
@Before("pointcut()")
public void runBefore() {
    System.out.println("执行了前置通知");
}
// 执行后置通知
@After("pointcut()")
public void runAfter() {
    System.out.println("执行了后置通知");
}

类似的@AfterReturning和@AfterThrowing写法一样,注解参数设置为上文设置的声明切点规则的函数名即可。

@Around注解除了遵守这个规则外,还需要设置返回值为Object类,参数设置为ProceedingJoinPoint类:

@Around("pointcut()")
public Object runAround(ProceedingJoinPoint joinPoint) throws Throwable {
    System.out.println("开始执行环绕方法");
    Object obj = null;
    // 执行拦截方法
    obj = joinPoint.proceed();
    System.out.println("结束执行环绕方法");
    return obj;
}

动态代理 

Spring AOP是基于动态代理技术实现的。调用者首先要通过代理才能到达目标对象,就比如说破在内陆访问chatgpt,就要先通过代理,代理就会把这些请求拦截下来,因此在内陆无法访问chatgpt等。动态代理包含两种方式:JDK动态代理和CGLib动态代理。

  • JDK动态代理:被代理类需要实现接口,然后通过反射机制,生成代理对象。
  • CGLib动态代理:被代理类可以不实现接口,通过继承被代理类的方式,生成代理对象。

Spring AOP默认使用JDK动态代理,如果没有实现接口就使用CGLib动态代理。

在Spring Boot2.x以后,默认使用CGLib动态代理。如果目标类实现了至少一个接口,就会优先考虑使用JDK动态代理。

可以在配置文件中设置属性为true强制使用CGLib代理(false强制使用JDK代理):

// 强制使用CGLib代理
spring.aop.proxy-target-class=true

织入

织入,把切面应用到目标对象并创建新的代理对象的过程,切面在指定的连接点被织入到目标对象中。简单来说就是动态代理的生成时机。织入可以在编译期、类加载期和运行期。Spring AOP是在运行期织入切面的。

你可能感兴趣的:(java,后端,spring,aop,spring,boot)