首页 > 其他分享 >Spring AOP 分享

Spring AOP 分享

时间:2023-05-12 18:44:06浏览次数:66  
标签:Spring Advice 代理 Bean AOP org 分享

初级篇

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,测试接口查看日志打印;
  • 展示切面打印非项目内的类方法执行日志,切换PointcutlogPointcut2()
  • 展示灵活使用配置,控制切面开启日志,开启@ConditionalOnProperty,修改yml文件;例如:测试环境要开启日志,生产环境要关闭日志,通过配置灵活控制。

PS:我们在此暂不考虑基于XmlApplicationContext系列;

怎么实现AOP?

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

  • 展示打印日志的静态代理,使用UserServiceStaticAopImpl注入Bean,测试接口查看日志打印;
  • 展示多种通知类型的静态代理,使用UserServiceStaticAopAnyImpl注入Bean,测试接口查看日志打印;

PS:我们在此暂不考虑基于XmlApplicationContext系列;

AOP的代码结构与核心概念

静态代理的AOP结构(简单易理解版):
image.png

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的应用需要开启<aop:aspectj-autoproxy />
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[index] + "=" + args[index])
                    .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("LogBeforeAdvice  execute {}", 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;
    }
}

运行代码后调试接口,看控制台打印输出:

: LogBeforeAdvice  execute 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画出他们的结构图如下:
Advised.png
UserService为例,Spring AOP生成的代理类大致结构:
image.png

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

时序图大致如下:
image.png

Advisor/Advice的执行顺序

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

  1. Advisororder值,没有则取Adviceorder值,还没有则为null,约等于最低优先级;
  2. 同一@Aspect类中定义多个Advice时,先按Around, Before, After, AfterReturning, AfterThrowing排序,再按方法名排序;
  3. 同一个Advice类实现多个Advice类型时,按 MethodInterceptorBeforeAdviceAfterReturningAdviceAfterThrowsAdvice排序;
  • 通过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的途径

ProxyConfig.png
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,
默认来自@EnableCachingorder属性,也是LOWEST_PRECEDENCE,可以直接在注解里修改。

Spring注解缓存@EnableCaching@Cacheable等的AOP支持。

BeanFactoryTransactionAttributeSourceAdvisor & TransactionInterceptor

排序:Ordered=Ordered.LOWEST_PRECEDENCE; // 继承自父类AbstractPointcutAdvisor,可set,
默认来自@EnableTransactionManagementorder属性,也是LOWEST_PRECEDENCE,可以直接在注解里修改。

Spring注解事务@Transactional的AOP支持。

高级篇

Spring自动创建AOP代理的时机

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

  • 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已经完成了 创建对象、属性填充、依赖注入、初始化方法的执行;

其它自动创建的还有ProxyFactoryBeanAbstractSingletonProxyFactoryBeanScopedProxyFactoryBean的派生类也会在初始化Bean时自动创建;

为什么BeanPostProcessor可能无法使用AOP的功能

Spring IOC容器在扫描所有的BeanDefinition后,会先按顺序初始化完所有的BeanPostProcessor,在初始化其它普通的Bean;而AbstractAdvisingBeanPostProcessorAbstractAutoProxyCreator这两个处理自动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代理类

  1. 首先AbstractAutoProxyCreator中,有三个Map对象保存AOP代理类的创建信息,分别是 targetSourcedBeansadvisedBeansearlyProxyReferences,当三者内存在Bean的代理对象时,则不会重复创建;
  2. AbstractAdvisingBeanPostProcessor中会判断 bean instanceof Advised,如果已经是AOP代理对象,则不会再创建新的代理对象,直接在此代理对象上添加Advisor

Spring AOP与常说的Spring三级缓存的关系

常说的Spring三级缓存指的是org.springframework.beans.factory.support.DefaultSingletonBeanRegistry类中的三个Map类型的属性,分别是:

Map<String, Object> singletonObjects; // 存放初始化完成的单例Bean
Map<String, Object> earlySingletonObjects; // 存放getEarlyBeanReference()钩子创建的早期Bean
Map<String, ObjectFactory<?>> singletonFactories; // 存放earlySingletonObject的工厂实例

IOC默认开启允许循环引用,但是在依赖注入时,但是Spring并不想把创建AOP代理的行为提前,因此设计了getEarlyBeanReference钩子和singletonFactoriesearlySingletonObjects,以支持循环引用。
在实例化Bean后,先创建出earlySingletonObject的工厂实例,以备后续依赖注入的其它Bean引用当前Bean,当真实存在循环依赖时,调用earlySingletonObject的工厂的方法构建早期Bean供其注入,并将这个早期Bean存放到earlySingletonObjects中,以备循环依赖不止一次;最终在当前Bean初始化完成后,移除早期Bean,将最终的Bean放入singletonFactories中,成为正式暴露的Bean实例;
org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#doCreateBean

// Eagerly cache singletons to be able to resolve circular references
// even when triggered by lifecycle interfaces like BeanFactoryAware.
boolean earlySingletonExposure = (mbd.isSingleton() && this.allowCircularReferences &&
        isSingletonCurrentlyInCreation(beanName));
if (earlySingletonExposure) {
    if (logger.isTraceEnabled()) {
        logger.trace("Eagerly caching bean '" + beanName +
                "' to allow for resolving potential circular references");
    }
    addSingletonFactory(beanName, () -> getEarlyBeanReference(beanName, mbd, bean));
}

org.springframework.beans.factory.support.AbstractBeanFactory#doGetBean

// Eagerly check singleton cache for manually registered singletons.
Object sharedInstance = getSingleton(beanName);

org.springframework.beans.factory.support.DefaultSingletonBeanRegistry#getSingleton(java.lang.String, boolean)

@Nullable
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
    // Quick check for existing instance without full singleton lock
    Object singletonObject = this.singletonObjects.get(beanName);
    if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
        singletonObject = this.earlySingletonObjects.get(beanName);
        if (singletonObject == null && allowEarlyReference) {
            synchronized (this.singletonObjects) {
                // Consistent creation of early reference within full singleton lock
                singletonObject = this.singletonObjects.get(beanName);
                if (singletonObject == null) {
                    singletonObject = this.earlySingletonObjects.get(beanName);
                    if (singletonObject == null) {
                        ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
                        if (singletonFactory != null) {
                            singletonObject = singletonFactory.getObject();
                            this.earlySingletonObjects.put(beanName, singletonObject);
                            this.singletonFactories.remove(beanName);
                        }
                    }
                }
            }
        }
    }
    return singletonObject;
}

扩展篇

ASM/Cglib动态字节码生成

篇幅太长,略

Spring缓存失效与AOP的瓜葛

Spring缓存切面主要依赖于AOP的实现,当对象内进行this.doSomething()调用时,因为没有经过代理对象,所以也没有缓存切面的功能;

Spring事务失效与AOP的瓜葛

与缓存切面一样,Spring事务切面主要依赖于AOP的实现,当对象内进行this.doSomething()调用时,因为没有经过代理对象,所以也没有事务切面的功能;

Synchronized和@Transactional 使用时,出现的问题

因为事务切面会在执行方法之前开启事务,之后再加锁,当锁住的代码执行完成后,在提交事务,
因此synchronized代码块执行是在事务之内执行的,可以推断在代码块执行完时,事务还未提交,其他线程进入synchronized代码块后,读取的库存数据不是最新的。
解决上面的方法,比较简单的可以在update方法之前加上synchronizedQ在还没有开事务之间就加锁,那么就可以保证线程同步

其他开发语言也可实现AOP

JavaScript的Proxy实现AOP

思考篇

AOP的优点

  • 降低了代码的耦合度:AOP可以将横切逻辑与业务逻辑分离,从而避免了代码的重复和冗余,提高了代码的可维护性和可读性。
  • 提高了代码的复用性:AOP可以将通用的横切逻辑抽象出来,作为一个切面,从而可以在不同的业务逻辑中重复使用,提高了代码的复用性。
  • 提高了系统的可扩展性:AOP可以动态地将横切逻辑织入到业务逻辑中,从而可以方便地添加新的功能,提高了系统的可扩展性。

AOP的缺点

  • AOP增加了系统的复杂度:AOP需要额外的配置和代码,增加了系统的复杂度,降低了系统的可读性和可维护性。
  • AOP可能会影响系统的性能:AOP需要在运行时进行额外的处理,可能会影响系统的性能,尤其是在大规模应用中。
  • AOP可能会引入新的问题:AOP可能会引入新的问题,如死锁、并发问题等,需要特别注意。

使用AOP的建议

  • 不要把应该与业务耦合的代码放到切面中
  • 注意切面的顺序
  • 注意this调用可能导致AOP失效

SpringBoot项目中使用AOP的最佳实践

标签:Spring,Advice,代理,Bean,AOP,org,分享
From: https://www.cnblogs.com/supalle/p/17396039.html

相关文章

  • 小白的一次Form组件封装分享
    Form组件介绍form表单常用在回显表单信息+验证表单+提交表单信息。通常包含输入框,选择框,日期选择框,文本框等可输入的组件。封装思路通过配置文件生成一个基本的表单,然后配合数据的双向绑定得到我们提交的数据,同时尽量保留第三方UI库组件提供的属性(Attributes)、插槽(Slots)和......
  • 资源分享系统-开放
    资源分享系统-开放提示:开源是学习的来源,分享是进步的体现近期,使用springboot结合layui写了一个技术管理系统,其实最主要是做一些资源的分享,可便捷生活和学习,包括免费听音乐的网站,免费看电影电视的网站做记录,和一部分分享的国内的chatgpt的网站供使用者访问。感兴趣的朋友可以进......
  • Spring表达式(Spring Expression Language,SpringEL)官方文档
    SpringEL是一个强大的表达式语言,支持在运行时查询和操作对象图。官方地址:https://docs.spring.io/spring-framework/docs/5.3.18/reference/html/core.html#expressions需要引入依赖:<dependency><groupId>org.springframework</groupId><artifactId>spring-expression</......
  • SpringBoot集成Jpa对数据进行排序、分页、条件查询和过滤
    之前介绍了SpringBoot集成Jpa的简单使用,接下来介绍一下使用Jpa连接数据库对数据进行排序、分页、条件查询和过滤操作。首先创建Springboot工程并已经继承JPA依赖,如果不知道可以查看我的另一篇文进行学习,这里不做介绍。文章地址(https://www.cnblogs.com/eternality/p/17391141.html......
  • Spring体系化笔记(韩顺平课程)
    SpringSpring核心学习内容IOC、AOP、JdbcTemplate、声明式事务1.Spring几个重要概念Spring可以整合其他的框架(Spring是管理框架的框架)Spring有两个核心的概念:IOC和AOPIOCInversionOfControl控制反转动态代理(学好了才能学好AOP)AOPAspect-ori......
  • TK Boss:分享就能赚钱,轻松开启全球电商新篇章
    在数字化时代,社交媒体的影响力已经无可忽视。现在,一款名为TKBoss的全新电商平台,利用社交媒体的力量,带给您全新的收益模式。TKBoss的模式十分简单:将TKBoss的产品链接分享至Facebook、Twitter、Instagram、TikTok等社交媒体,当有人通过您的链接购买商品,您就能获取相应的佣金。无需......
  • 分享Python采集88个html5代码,总有一款适合您
    分享Python采集88个html5代码,总有一款适合您Python采集的88个html5代码下载链接:https://pan.baidu.com/s/1wJzpYBlHIGtNgTisJwCQ6Q?pwd=yr96提取码:yr96imgplay-基于canvas的图片序列播放jQuery插件基于HTML5Canvas和Rebound动画的Loading加载动画特效超酷创意分段式SVG文字动画特......
  • 【经验分享】最新Edge游览器游览Flash网页的办法
    环境:工具:MicrosoftEdgeDev系统版本:Windows10需求描述:描述:需要正常游览Flash页面的所有内容,不能有缺少的动画提示:*本文章是最简单的且直接的,若你的电脑不适合这个游览器,可以看我另外一篇文章。https://www.cnblogs.com/boluo0423/p/17358646.html提示:若按照教程还......
  • [SpringCloud]Spring-Cloud-Gateway之启动过程(源码分析)
    1前言1.1环境信息Spring-Cloud-Gateway:2.2.9.RELEASEorg.springframework.boot:spring-boot:2.3.12.RELEASEio.projectreactor.netty:reactor-netty:0.9.20.RELEASEio.netty:netty-transport:4.1.65.FINAL2启动过程#与Netty的调用链路2.1简版(V1.0)cn.seres.b......
  • 国内免费ChatGPT镜像网站分享,稳定高效,回复速度快
    一、网站效果图二、网站简介:百晓生,专注AI,让未来触手可及!中文免费版ChatGPT,解放你的脑力与劳动力。支持AI创作,AI绘画,角色扮演等,更多好玩技巧等你来发现!三、网站地址:https://ai.hxkj.vip四、使用体验:1、首先来看看经典的“一棵树上有10只鸟,打死一只之后,树上还剩几只”问......