一,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,走剩下的流程。