饭宝 发表于 2023-5-12 18:31:34

Spring AOP 分享

初级篇

AOP是什么?

Aspect-oriented Programming (AOP) 即面向切面编程。
简单来说,AOP 是一种编程范式,允许我们模块化地定义横跨多个对象的行为。AOP 可以帮助我们将应用程序的关注点分离,使得代码更加清晰、易于维护和扩展。
大白话:在方法执行前后运行指定代码,比如日志记录、事务开启/提交/回滚等。
为什么要AOP?

AOP可以帮助我们解决在代码中耦合度高的问题,让我们的代码更加模块化和易于维护。
具体来说,AOP可以通过在运行时动态地将通用功能(例如日志记录、性能分析、事务管理)应用于多个模块,而无需修改它们的代码。这样可以避免代码重复和嵌套,提高代码的复用性和可维护性。
另外,在复杂的业务场景中,多个模块可能需要共享某些共同的功能,而AOP可以让这些功能从模块中抽离出来,以便更好地进行组织和重用。
总的来说,AOP可以让我们更好地实现代码的分离和聚合,从而获得更高效、更可靠的代码。
大白话:增强原方法的功能,解耦通用功能,透明化静默操作。
举例 事务切面切面:
增强原方法的功能:原本方法使用的是数据库连接默认的策略自动提交事务的,有了切面能够保证方法内同一事务了;
解耦通用功能:但是很多方法都需要做事务的控制,有了切面不需要我们每一个方法都加几行相同的代码;
透明化静默操作:方法本身需要知道我怎么开的事务?需要知道我什么时候多打印了日志吗?


[*]伪代码:没有使用AOP前,每个方法都要CV一遍打印方法执行日志
@Override
public UserPO findByUsername(@AutoTrim String username) {
    log.info("execute findByUsername by username={}", username);
    Optional<UserPO> opt = userRepository.findOne((root, query, criteriaBuilder) -> criteriaBuilder.equal(root.get("username"), username));
    Assert.isTrue(opt.isPresent(), "没有找到用户");
    UserPO userPO = opt.get();
    log.info("execute findByUsername by username={}; return {} ", username, userPO);
    return userPO;
}

@Override
public UserPO findByEmail(@AutoTrim String email) {
    log.info("execute findByEmail by email={}", email);
    Optional<UserPO> opt = userRepository.findOne((root, query, criteriaBuilder) -> criteriaBuilder.equal(root.get("email"), email));
    Assert.isTrue(opt.isPresent(), "没有找到用户");
    UserPO userPO = opt.get();
    log.info("execute findByEmail by email={}; return {} ", email, userPO);
    return userPO;
}

[*]伪代码:使用AOP后,无需关心方法执行日志的打印
@Override
public UserPO findByUsername(@AutoTrim String username) {
    Optional<UserPO> opt = userRepository.findOne((root, query, criteriaBuilder) -> criteriaBuilder.equal(root.get("username"), username));
    Assert.isTrue(opt.isPresent(), "没有找到用户");
    return opt.get();
}

@Override
public UserPO findByEmail(@AutoTrim String email) {
    Optional<UserPO> opt = userRepository.findOne((root, query, criteriaBuilder) -> criteriaBuilder.equal(root.get("email"), email));
    Assert.isTrue(opt.isPresent(), "没有找到用户");
    return opt.get();
}此处代码演示 【part1】
演示内容:

[*]展示未使用AOP前,使用UserServiceImpl注入Bean,测试接口查看日志打印;
[*]展示使用AOP后,使用UserServiceAopImpl注入Bean,测试接口查看日志打印;
[*]展示切面打印非项目内的类方法执行日志,切换Pointcut为logPointcut2();
[*]展示灵活使用配置,控制切面开启日志,开启@ConditionalOnProperty,修改yml文件;例如:测试环境要开启日志,生产环境要关闭日志,通过配置灵活控制。
PS:我们在此暂不考虑基于XmlApplicationContext系列;
怎么实现AOP?

AOP的实现依赖于多态和动态代理。
为了更好的理解,我们可以先举例一个静态代理的类来分析。
此处代码演示 【part2】
演示内容:

[*]展示打印日志的静态代理,使用UserServiceStaticAopImpl注入Bean,测试接口查看日志打印;
[*]展示多种通知类型的静态代理,使用UserServiceStaticAopAnyImpl注入Bean,测试接口查看日志打印;
PS:我们在此暂不考虑基于XmlApplicationContext系列;
AOP的代码结构与核心概念

静态代理的AOP结构(简单易理解版):
https://cdn.nlark.com/yuque/0/2023/png/21749835/1683793011608-6e29643e-cbda-4218-a625-7b68d2c97587.png#averageHue=%23f9f9f8&clientId=u66b4ae87-5f46-4&from=paste&id=u5a0ea771&originHeight=984&originWidth=1566&originalType=url&ratio=1.5&rotation=0&showTitle=false&size=126272&status=done&style=none&taskId=u8dbcc8c4-42d9-4d0a-b216-a3a7a03a53b&title=
Aspect

切面:指横跨多个类的一个关注点,它与不同类中相似的方法相对应。如保存数据时添加日志,可以新建一个切面配置日志记录逻辑。
大白话:一个模块化的切面程序,也可以理解为是一个实现切面功能的类;
用@Aspect定义的Bean Class,或者Spring xml配置里的aop:aspect标签:
<aop:config>
<aop:aspect id="myAspect" ref="aBean">
    ...
</aop:aspect>
</aop:config>Join Point

连接点:指在应用程序执行过程中的某个特定位置,如方法调用或异常处理等。
可以理解为要切面的对象类型,比如要加切面的目标是构造器,或者一个方法,或者是一个属性的赋值;
AspectJ中可以有很多种,详见AspectJ Join Points;
SpringAOP中只有一种,就是方法执行(Method execution);
Advice

通知:指在切面的某个连接点上执行的代码。通知有许多类型,包括“前置通知”、“后置通知”、“返回通知”、“异常通知”、“环绕通知”,其中“环绕通知”能够完全控制目标方法的执行。
Pointcut

切入点:指一个或多个连接点,通常定义在一个正则表达式中,描述哪些方法会被拦截。
参考:
Join Points and Pointcuts
Spring 之AOP AspectJ切入点语法详解
例:
within(com.supalle.springaop..*.UserServiceAopImpl)
execution(* com.supalle.springaop.*.*(..))
within(com.supalle.springaop.TestController) && @annotation(com.supalle.springaop.Log)
其它


[*]Introduction
[*]Target
[*]AOP proxy
[*]Weaving
详见:AOP Concepts
Spring AOP常用的方式

此处代码演示 【part3】
@Aspect

XmlApplicationContext的应用需要开启
AnnotationConfigApplicationContext的应用需要开启@EnableAspectJAutoProxy;SpringBoot环境下可以不用,在 org.springframework.boot.autoconfigure.aop.AopAutoConfiguration里已经默认启用;
package com.supalle.springaop.aspect;

import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.*;
import org.aspectj.lang.reflect.MethodSignature;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

import java.util.stream.Collectors;
import java.util.stream.IntStream;

/**
* 日志切面
*
* @author supalle
* @see {@link https://docs.spring.io/spring-framework/docs/current/reference/html/core.html#aop-pointcuts}
*/
@Slf4j
@Aspect
@Component
//@ConditionalOnProperty(value = "log-execution", havingValue = "true")
public class LogAspect {

    @Pointcut("within(com.supalle.springaop..*.UserServiceAopImpl)")
    // the pointcut within *AopImpl
    private void logPointcut() {
    }

    @Pointcut("within(com.supalle.springaop..*.UserServiceAopImpl) || within(com.zaxxer.hikari.HikariDataSource)")
    private void logPointcut2() {
    }

    @Before("logPointcut()")
    public void beforeLog(JoinPoint joinPoint) {
      log.info("Before Advice 1 execute {}", joinPoint.getSignature().getName());
    }

    // 同一个Aspect内,@Order排序无效,需要靠方法名排序,比如把当前方法改为beforeLog2,就能运行在beforeLog前面
    @Before("logPointcut()")
    public void beforeLog2(JoinPoint joinPoint) {
      log.info("Before Advice 2 execute {}", joinPoint.getSignature().getName());
    }

    @AfterReturning(value = "logPointcut()", returning = "returnValue")
    public void afterReturningLog(JoinPoint joinPoint, Object returnValue) {
      log.info("AfterReturning Advice execute {}; {} ", joinPoint.getSignature().getName(), returnValue);
    }

    @AfterThrowing(value = "logPointcut()", throwing = "throwable")
    public void afterThrowingLog(JoinPoint joinPoint, Throwable throwable) {
      log.info("AfterThrowing Advice execute {}; {}", joinPoint.getSignature().getName(), throwable.getMessage());
    }

    @After("logPointcut()")
    public void afterLog(JoinPoint joinPoint) {
      log.info("After Advice execute {}", joinPoint.getSignature().getName());
    }

    @Around("logPointcut()")
    public Object aroundLog(ProceedingJoinPoint joinPoint) throws Throwable {
      Signature joinPointSignature = joinPoint.getSignature();
      String name = joinPointSignature.getName();
      Logger logger = LoggerFactory.getLogger(joinPointSignature.getDeclaringTypeName());

      Object[] args = joinPoint.getArgs();
      String[] argNames = ((MethodSignature) joinPointSignature).getParameterNames();
      String argString = "";
      if (args != null && args.length > 0) {
            argString = " by " + IntStream.range(0, args.length)
                  .mapToObj(index -> argNames + "=" + args)
                  .collect(Collectors.joining(" , "));
      }

      logger.info("Around Advice execute {}{}", name, argString);
      Object returnValue = joinPoint.proceed();
      logger.info("Around Advice execute {}{}; return {} ", name, argString, returnValue);

      return returnValue;
    }

}XML配置


Advisor

org.springframework.aop.framework.autoproxy.AbstractAdvisorAutoProxyCreator#findEligibleAdvisors中会查找容器中所有的Advisor类型Bean,也属于AutoProxy的支持;


[*]LogBeforeAdvice:定义一个Advice,加@Component注册为Bean
package com.supalle.springaop.advice;

import lombok.extern.slf4j.Slf4j;
import org.springframework.aop.MethodBeforeAdvice;
import org.springframework.stereotype.Component;

import java.lang.reflect.Method;

/**
* org.springframework.aop.framework.adapter.DefaultAdvisorAdapterRegistry的支持
*/
@Slf4j
@Component
public class LogBeforeAdvice implements MethodBeforeAdvice {

    @Override
    public void before(Method method, Object[] args, Object target) throws Throwable {
      log.info("LogBeforeAdviceexecute {}", method.getName());
    }

}

[*]LogBeforeAdvisor:定义一个Advisor,加@Component注册为Bean
package com.supalle.springaop.advisor;

import com.supalle.springaop.advice.LogBeforeAdvice;
import lombok.RequiredArgsConstructor;
import org.aopalliance.aop.Advice;
import org.springframework.aop.ClassFilter;
import org.springframework.aop.MethodMatcher;
import org.springframework.aop.Pointcut;
import org.springframework.aop.support.AbstractPointcutAdvisor;
import org.springframework.stereotype.Component;

@Component
@RequiredArgsConstructor
public class LogBeforeAdvisor extends AbstractPointcutAdvisor {
    private final LogBeforeAdvice advice;

    @Override
    public Pointcut getPointcut() {
      Pointcut pointcut = new Pointcut() {
            @Override
            public ClassFilter getClassFilter() {
                return clazz -> "UserServiceAopImpl".equals(clazz.getSimpleName());
            }

            @Override
            public MethodMatcher getMethodMatcher() {
                return MethodMatcher.TRUE;
            }
      };
      return pointcut;
    }

    @Override
    public Advice getAdvice() {
      return advice;
    }
}运行代码后调试接口,看控制台打印输出:
: LogBeforeAdviceexecute findByUsername程序员创建

用的不多,可以参考Programmatic Creation of @AspectJ Proxies
进阶篇

Spring AOP与动态代理

动态代理

动态代理是指在程序运行时动态生成代理类的技术,即不需要手工编写代理类的源代码,而是在程序运行期间通过反射等机制动态地生成。
动态代理模式可以帮助我们减少重复的代码,提高代码的可维护性和可扩展性。通常情况下,我们都是通过实现接口来创建代理对象,但是如果一个类没有实现任何接口,我们仍然可以通过动态代理来创建代理对象。动态代理模式适用于一些横切关注点(cross-cutting concerns)的处理,例如日志、安全、事务等功能。
在 Java 中,动态代理模式主要有两种实现方式:
基于 JDK 动态代理(JDK Dynamic Proxy):JDK 提供了一个 java.lang.reflect.Proxy 类,可以动态地创建实现一组给定接口的代理类。要求被代理对象必须实现至少一个接口,并通过 Proxy 类的静态方法 newProxyInstance 来创建代理对象。
基于 CGLIB 动态代理:CGLIB(Code Generation Library) 是一个基于 ASM(Java 字节码操作框架)的高性能字节码生成库,可以在运行时动态生成字节码,并生成对应的代理类。要求被代理对象必须有默认构造函数,并通过 CGLIB 库提供的代理工厂(Enhancer 类)来创建代理对象。
总之,动态代理模式可以帮助我们在程序运行期间动态地生成代理类,从而达到增强功能、添加处理逻辑等目的。
Spring AOP

Spring AOP 就是动态代理的一种落地实现,底层是通过JDK Proxy和Cglib&ASM策略来进行动态生成代理类;
Spring Native应用下实现AOP的话,刚出来的时候我看过一遍,因为无法动态生成字节码,所以是在AOT编译时,给应用中所有的类都生成了一个代理类,当这个类在实际运行时需要动态代理时,则加载这个代理类直接使用。(时过境迁,现在不知道实现方式变了没有)
AOT(Ahead-of-time )compiler:预先编译器;
JIT(Just-In-Time)compiler:即时编译器;
Spring AOP的结构与核心类

除了上文说的AOP基本概念在Spring中有对应的类外,Spring AOP APIs中还有其它几个核心类,用于具体的AOP实现。

[*]Advised:代理类的顶层接口,包含委托对象和其信息,以及应用的Advisor的集合;
[*]Advisor:一个Pointcut+ Advice的组合;
[*]AopProxy:AOP具体的代理实现,可能是JDK Proxy,也可能是Cglib Proxy;


[*]JdkDynamicAopProxy
[*]ObjenesisCglibAopProxy
org.springframework.aop.framework.DefaultAopProxyFactory#createAopProxy
用UML画出他们的结构图如下:
https://cdn.nlark.com/yuque/0/2023/png/21749835/1683853946557-c2593e03-99ed-42fe-a3bf-65d32da6931e.png#averageHue=%232e2e2e&clientId=ue03ea021-1d3c-4&from=paste&height=848&id=uedb83239&originHeight=1272&originWidth=1406&originalType=binary&ratio=1.5&rotation=0&showTitle=false&size=88080&status=done&style=none&taskId=ufab41f4a-7b45-4d83-93ae-518a3e8657e&title=&width=937.3333333333334
以UserService为例,Spring AOP生成的代理类大致结构:
https://cdn.nlark.com/yuque/0/2023/png/21749835/1683855263686-827087b2-c93f-408c-8da8-09e91cb60778.png#averageHue=%23707070&clientId=ue03ea021-1d3c-4&from=paste&id=u69c02bc0&originHeight=1896&originWidth=3072&originalType=url&ratio=1.5&rotation=0&showTitle=false&size=402595&status=done&style=none&taskId=u29da1cc5-ea3d-4d0c-ae4b-42ecf87bea2&title=
Spring AOP的调用链

仔细思考,上文举例的静态代理实现AOP的代码,是不够健全的,尤其是在Advice的顺序上;
此处代码演示 【part2】:beforeAspects切面中添加 System.out.println(1/0);使其抛出异常
引用Spring官方文档的原文:
Advice: Action taken by an aspect at a particular join point. Different types of advice include "around", "before", and "after" advice. (Advice types are discussed later.) Many AOP frameworks, including Spring, model an advice as an interceptor and maintain a chain of interceptors around the join point.
这句“Many AOP frameworks, including Spring, model an advice as an interceptor and maintain a chain of interceptors around the join point.”翻译过来就是“包括Spring在内的许多AOP框架都将Advice建模为拦截器,并在连接点周围维护拦截器链。”
很好,使用类似于javax.servlet.Filter拦截器的方式,就可以很好的控制Advice的执行顺序。
详见:org.springframework.aop.framework.AdvisedSupport#getInterceptorsAndDynamicInterceptionAdvice
时序图大致如下:
https://cdn.nlark.com/yuque/0/2023/png/21749835/1683857965948-0c925936-be3c-4b61-9246-af3872397354.png#averageHue=%23fafafa&clientId=u9ac1d0c7-8537-4&from=paste&id=ud82ffd7b&originHeight=821&originWidth=1211&originalType=url&ratio=1.5&rotation=0&showTitle=false&size=64697&status=done&style=none&taskId=uf2dfa273-f376-44a1-9d7e-1c1e9f0d6e7&title=
Advisor/Advice的执行顺序

调用链的顺序,基于Advised中的advisors集合顺序,主要的排序策略如下:

[*]Advisor的order值,没有则取Advice的order值,还没有则为null,约等于最低优先级;
[*]同一@Aspect类中定义多个Advice时,先按Around, Before, After, AfterReturning, AfterThrowing排序,再按方法名排序;
[*]同一个Advice类实现多个Advice类型时,按 MethodInterceptor,BeforeAdvice,AfterReturningAdvice,AfterThrowsAdvice排序;


[*]通过AbstractAdvisingBeanPostProcessor的后置处理器添加的Advisor,后置处理器可以通过beforeExistingAdvisors属性控制是否排序在所有已经存在的Advisor前;


[*]Spring官方文档:Advice Ordering
[*]排序Advisors:org.springframework.aop.aspectj.autoproxy.AspectJAwareAdvisorAutoProxyCreator#sortAdvisors
[*]排序同一Aspect中的Advice:org.springframework.aop.aspectj.annotation.ReflectiveAspectJAdvisorFactory#getAdvisorMethods
PS:如果Advisor/Advice是实现了Ordered接口的话,@Order注解是不生效的。
Spring AOP添加Advisor的途径

https://cdn.nlark.com/yuque/0/2023/png/21749835/1674015819683-4049bd6b-afe3-4662-a197-4cb50ff358e0.png#averageHue=%232d2d2d&clientId=u12eeec89-5cac-4&from=drop&id=u9a3661a7&originHeight=1656&originWidth=5421&originalType=binary&ratio=1&rotation=0&showTitle=false&size=144779&status=done&style=none&taskId=u9cfc25d5-60be-4ce9-9fd3-f6edcae0bb8&title=
ProxyConfig下分几个家族,都可以添加Advisor,创建AOP代理类。
ProxyProcessorSupport家族

AbstractAdvisingBeanPostProcessor

AbstractAutoProxyCreator

AdvisedSupport家族

ProxyFactoryBean

AbstractSingletonProxyFactoryBean家族

ScopedProxyFactoryBean家族

常见的Advisor & Advice

AsyncAnnotationAdvisor & AsyncExecutionInterceptor

排序:Ordered=Ordered.HIGHEST_PRECEDENCE; // Integer.MIN_VALUE;方法内写死,继承才可以修改
注意:AsyncAnnotationAdvisor是在AsyncAnnotationBeanPostProcessor后置处理器中创建和匹配,beforeExistingAdvisors=true;排序在所有已经存在的Advisor前。
Spring异步执行的AOP支持。
AsyncAnnotationAdvisor & AnnotationAsyncExecutionInterceptor

排序:Ordered=Ordered.HIGHEST_PRECEDENCE; // 继承自父类AsyncExecutionInterceptor
Spring注解@Async异步执行的AOP支持。
RetryConfiguration & AnnotationAwareRetryOperationsInterceptor

排序:Ordered=Ordered.LOWEST_PRECEDENCE; // 继承自父类AbstractPointcutAdvisor,可set;
注意:RetryConfiguration实现IntroductionAdvisor,排序优先于同order的未实现IntroductionAdvisor的Advisor。
Spring注解@Retryable方法失败重试的AOP支持。
BeanFactoryCacheOperationSourceAdvisor & CacheInterceptor

排序:Ordered=Ordered.LOWEST_PRECEDENCE; // 继承自父类AbstractPointcutAdvisor,可set,
默认来自@EnableCaching的order属性,也是LOWEST_PRECEDENCE,可以直接在注解里修改。
Spring注解缓存@EnableCaching、@Cacheable等的AOP支持。
BeanFactoryTransactionAttributeSourceAdvisor & TransactionInterceptor

排序:Ordered=Ordered.LOWEST_PRECEDENCE; // 继承自父类AbstractPointcutAdvisor,可set,
默认来自@EnableTransactionManagement的order属性,也是LOWEST_PRECEDENCE,可以直接在注解里修改。
Spring注解事务@Transactional的AOP支持。
高级篇

Spring自动创建AOP代理的时机

Spring AOP代理类创建的时机,主要是在Bean被引用之前,依靠BeanPostProcessor来实现;
这就要靠ProxyProcessorSupport家族的AbstractAdvisingBeanPostProcessor和AbstractAutoProxyCreator了。

[*]AbstractAdvisingBeanPostProcessor对单个Advisor进行AOP切面的添加;
[*]AbstractAutoProxyCreator对所有满足Pointcut的Bean进行切面的添加;
两者大部分情况会在postProcessAfterInitialization()钩子方法中创建Bean的AOP代理对象,但也有一些特殊情况,比如自定义TargetSource的时候,会在postProcessBeforeInstantiation()钩子方法中创建代理对象;
还有一种情况是在getEarlyBeanReference()钩子方法时创建,也会提前创建好AOP代理对象;
详见源码:

[*]org.springframework.aop.framework.AbstractAdvisingBeanPostProcessor#postProcessAfterInitialization
[*]org.springframework.aop.framework.autoproxy.AbstractAutoProxyCreator#postProcessAfterInitialization
另外注释一下:postProcessBeforeInstantiation()钩子执行时,Bean已经完成了 创建对象、属性填充、依赖注入、初始化方法的执行;
其它自动创建的还有ProxyFactoryBean、AbstractSingletonProxyFactoryBean、ScopedProxyFactoryBean的派生类也会在初始化Bean时自动创建;
为什么BeanPostProcessor可能无法使用AOP的功能

Spring IOC容器在扫描所有的BeanDefinition后,会先按顺序初始化完所有的BeanPostProcessor,在初始化其它普通的Bean;而AbstractAdvisingBeanPostProcessor和AbstractAutoProxyCreator这两个处理自动AOP的后置处理器排序都是 Ordered=LOWEST_PRECEDENCE,因此,在这两后置处理器之前初始化的Bean都无法被他们进行自动AOP代理。
Customizing Beans by Using a BeanPostProcessor
Spring官方提示:
BeanPostProcessor instances and AOP auto-proxying
Classes that implement the BeanPostProcessor interface are special and are treated differently by the container. All BeanPostProcessor instances and beans that they directly reference are instantiated on startup, as part of the special startup phase of the ApplicationContext. Next, all BeanPostProcessor instances are registered in a sorted fashion and applied to all further beans in the container. Because AOP auto-proxying is implemented as a BeanPostProcessor itself, neither BeanPostProcessor instances nor the beans they directly reference are eligible for auto-proxying and, thus, do not have aspects woven into them.
源码详见:org.springframework.context.support.PostProcessorRegistrationDelegate#registerBeanPostProcessors
Spring如何防止反复创建AOP代理类


[*]首先AbstractAutoProxyCreator中,有三个Map对象保存AOP代理类的创建信息,分别是 targetSourcedBeans、advisedBeans、earlyProxyReferences,当三者内存在Bean的代理对象时,则不会重复创建;
[*]AbstractAdvisingBeanPostProcessor中会判断 bean instanceof Advised,如果已经是AOP代理对象,则不会再创建新的代理对象,直接在此代理对象上添加Advisor;
Spring AOP与常说的Spring三级缓存的关系

常说的Spring三级缓存指的是org.springframework.beans.factory.support.DefaultSingletonBeanRegistry类中的三个Map类型的属性,分别是:
Map singletonObjects; // 存放初始化完成的单例BeanMap earlySingletonObjects; // 存放getEarlyBeanReference()钩子创建的早期BeanMap
页: [1]
查看完整版本: Spring AOP 分享