Spring的三级缓存与循环依赖

一,spring4版本与spring5版本aop的执行顺序

依赖

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.5.9.RELEASE</version>
        <!--		<version>2.2.6.RELEASE</version>-->
        <relativePath/>
    </parent>
    <groupId>com.yhd</groupId>
    <artifactId>aop</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>aop</name>
    <description>Demo project for Spring Boot</description>

    <properties>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

被增强方法

/**
 * @author yhd
 * @createtime 2020/11/27 11:37
 */
@Service
public class Demo {

    public int dev(int x,int y){
        System.out.println(x/y);
        return x/y;
    }
}

aop切面类

/**
 * @author yhd
 * @createtime 2020/11/27 11:01
 */
@Component
@Aspect
public class AopDemo {

    @Pointcut(value = "execution(public int com.yhd.aop.*.*.*(..))")
    public void  pointCut(){

    }
    @Before("pointCut()")
    public void before(JoinPoint joinPoint){
        System.out.println("前置通知。。。");
    }
    @AfterReturning("pointCut()")
    public void afterReturing(){
        System.out.println("返回通知。。。");
    }
    @AfterThrowing("pointCut()")
    public void afterThrowing(){
        System.out.println("异常通知。。。");
    }
    @After("pointCut()")
    public void after(){
        System.out.println("最终通知。。。");
    }
    @Around("pointCut()")
    public int around(ProceedingJoinPoint joinPoint)throws Throwable{

            System.out.println("环绕通知----前置通知");
            Object proceed = joinPoint.proceed();
            System.out.println("环绕通知---后置通知");
            return (int) proceed;

    }
}

测试类

/**
 * @author yhd
 * @createtime 2020/11/27 11:01
 */
@Component
@Aspect
public class AopDemo {

    @Pointcut(value = "execution(public int com.yhd.aop.*.*.*(..))")
    public void  pointCut(){

    }
    @Before("pointCut()")
    public void before(JoinPoint joinPoint){
        System.out.println("前置通知。。。");
    }
    @AfterReturning("pointCut()")
    public void afterReturing(){
        System.out.println("返回通知。。。");
    }
    @AfterThrowing("pointCut()")
    public void afterThrowing(){
        System.out.println("异常通知。。。");
    }
    @After("pointCut()")
    public void after(){
        System.out.println("最终通知。。。");
    }
    @Around("pointCut()")
    public int around(ProceedingJoinPoint joinPoint)throws Throwable{

            System.out.println("环绕通知----前置通知");
            Object proceed = joinPoint.proceed();
            System.out.println("环绕通知---后置通知");
            return (int) proceed;

    }
}

最终结论

在这里插入图片描述

二,spring利用三级缓存来解决循环依赖

什么是循环依赖?

多个bean之间相互依赖,形成了一个闭环。

@Component
public class A {
    @Autowired
    private B b;
}

@Component
class B {
    @Autowired
    private C c;
}

@Component
class C {
    @Autowired
    private A a;
}

通常来说,谈到spring容器如何解决循环依赖,一定是指默认的单利bean,属性互相引用的场景。

也就是说,spring的循环依赖,是spring容器注入时候出现的问题。

spring容器循环依赖报错BaanCurrentlylnCreationException

1.构造器方式注入依赖

构造器方式注入没有办法解决循环依赖,想用构造器注入解决循环依赖是不存在的。

    @Test
    public void test2(){
        new A(new B(new A(。。。)));
    }

2.set方法注入依赖

set方法是可以解决循环依赖注入问题的

@Component
@Setter
public class A {
    
    private B b;
}

@Component
@Setter
public class B {

    private A a;
}

    @Test
    public void test2(){
        A a = new A();
        B b = new B();
        a.setB(b);
        b.setA(a);
    }

3.spring容器依赖注入

默认的单例场景是支持循环依赖的,不报错。

多例场景是不支持循环依赖的,报错。

依赖
    <!-- 统一管理jar包版本 -->
    <properties>
        <spring.version>5.0.2.RELEASE</spring.version>
        <slf4j.version>1.6.6</slf4j.version>
        <log4j.version>1.2.12</log4j.version>
        <shiro.version>1.2.3</shiro.version>
        <mysql.version>5.1.6</mysql.version>
        <mybatis.version>3.4.5</mybatis.version>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <maven.compiler.source>1.8</maven.compiler.source>
        <maven.compiler.target>1.8</maven.compiler.target>
        <spring.security.version>5.0.1.RELEASE</spring.security.version>
    </properties>

    <!-- 锁定jar包版本 -->
    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-context</artifactId>
                <version>${spring.version}</version>
            </dependency>
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-web</artifactId>
                <version>${spring.version}</version>
            </dependency>
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-webmvc</artifactId>
                <version>${spring.version}</version>
            </dependency>
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-tx</artifactId>
                <version>${spring.version}</version>
            </dependency>
            <dependency>
                <groupId>org.springframework</groupId>
                <artifactId>spring-test</artifactId>
                <version>${spring.version}</version>
            </dependency>
            <dependency>
                <groupId>org.mybatis</groupId>
                <artifactId>mybatis</artifactId>
                <version>${mybatis.version}</version>
            </dependency>
        </dependencies>
    </dependencyManagement>

    <!-- 项目依赖jar包 -->
    <dependencies>
        <!-- spring -->
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.6.8</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-aop</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context-support</artifactId>
            <version>${spring.version}</version>
        </dependency>

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-orm</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-beans</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-test</artifactId>
            <version>${spring.version}</version>
        </dependency>

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-tx</artifactId>
            <version>${spring.version}</version>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.16</version>
        </dependency>
    </dependencies>
beans.xml
<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
    <context:component-scan base-package="com"></context:component-scan>
</beans>
配置类
@Configuration
public class SpringConfig {

    @Bean
    @Scope
    public A a(){
        return new A();
    }
    @Bean
    @Scope
    public B b(){
        return new B();
    }
}
实体类
@Setter
public class A {

    private B b;
}

@Setter
public class B {

    private A a;
}

4.结论

spring内部通过三级缓存来解决循环依赖,所谓的三级缓存,其实就是spring容器内部用来解决循环依赖问题的三个map。

只有单例的bean会通过三级缓存提前暴露来解决循环依赖的问题,而非单利的bean,每次从容器中获取的都是一个新的对象,都会重新创建,所以非单利的bean是没有缓存的,不会讲其放到三级缓存中。

实例化:堆空间申请一块空间。

初始化:完成属性填充赋值。
在这里插入图片描述

源码追踪结论

整个执行流程主要是三个缓存map和四个方法

	//一级缓存:实例化完的bean
	private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);

	//三级缓存:单例bean工厂
	private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);

	//二级缓存:早期暴露的bean
	private final Map<String, Object> earlySingletonObjects = new HashMap<>(16);

四个方法分别是

getSingleton 希望从容器里面获取单例的bean

doCreateBean 没有就创建bean

populateBean 创建以后填充属性

addSingleton 填充完成在添加到容器使用

AB俩对象在三级缓存中的迁移说明

1.A创建过程中需要B,于是A将自己放到三级缓存去实例化B

2.B实例化的时候发现需要A,于是B先查一级缓存,没有,再查二级缓存,没有,再查三级缓存,找到了A,然后把三级缓存里面的A放到二级缓存里面,并删除三级缓存里面的A。

3.B顺利初始化完毕,将自己放到一级缓存里面(此时B里面的A还未创建完),然后接着回来创建A,此时B已经创建结束,直接从一级缓存里面拿到B,然后完成创建,并将A自己放到一级缓存里面。

spring解决循环依赖依靠的是Bean的中间态这个概念,而这个中间态指的是已经实例化但还没初始化的状态----->半成品。

实例化的过程又是通过构造器创建的,如果A还没创建出来就不能提前曝光,所以构造器的循环依赖无法解决。

spring为了解决单利的循环依赖问题,使用了三级缓存:其中一级为单例池,二级为提前曝光对象,三级为提前曝光对象工厂。

假设A,B循环引用,实例化A的时候就讲其放入到三级缓存中,接着填充属性的时候,发现依赖了B,同样的路程也是实例化后放入三级缓存,接着去填充属性时又发现自己依赖A,这时候从缓存中查找到早期暴露的A,没有AOP代理的话,直接将A的原始对象注入B,完成B的初始化后,进行属性填充和初始化,这时候B完成后,就去完成剩下的A的步骤,如果有AOP代理,就进行AOP处理获取代理后的对象A,,注入B,走剩下的流程。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值