Spring的三级缓存和SpringMVC的流程

目录

前言

一、Spring的三级缓存

[1. 循环依赖](#1. 循环依赖)

[2. Spring 不支持的循环依赖](#2. Spring 不支持的循环依赖)

[3. 解决循环依赖](#3. 解决循环依赖)

[二、SpringMVC 的流程](#二、SpringMVC 的流程)


前言

Spring通过三级缓存机制解决循环依赖问题:一级缓存存储完整Bean,二级缓存存储半成品Bean,三级缓存存储Bean工厂类。该机制能处理普通循环依赖,但无法解决构造器循环依赖、原型Bean循环依赖和复杂AOP场景下的循环依赖。Spring MVC处理流程包括:请求分发、Handler查找、处理执行、视图解析、视图渲染和结果返回六个步骤。


一、Spring的三级缓存

1. 循环依赖

循环依赖就是 A 对象完成实例化,需要注入 B 对象,B 对象完成实例化,也需要注入 A 对象;只要产生了依赖的闭环,就产生了循环依赖;

Bean 的生命周期分为 5 个步骤,分别是:实例化,属性注入,初始化,使用以及销毁;

在属性注入这个步骤有可能会有循环依赖的问题;

Spring 保存 Bean 是采用缓存的方式,使用一个 Map<String, Object> 的数据结构,key 是 Bean 的名称,value 是 Bean 对象,需要时从缓存中获取;

如果 A 对象和 B 对象互相依赖:

A 进行属性注入的时候,发现缓存中没有 B 对象,于是实例化 B 对象;

实例化 B 对象,进行属性注入的时候,发现缓存中也没有 A 对象,于是实例化 A 对象,于是又重复上面的步骤;

2. Spring 不支持的循环依赖

Spring 能解决循环依赖问题,但也不是能解决全部的情况;

  1. 构造器循环依赖:当两个 Bean 在构造器中循环依赖,会抛出异常 BeanCurrentlyInCreationException;

  2. 原型 Bean 循环依赖:原型作用域的 Bean,生命周期是短暂的,每次使用都会创建一个新的 Bean,因此 Spring 容器中不缓存这样的 Bean,也不能解决循环依赖的问题;

  3. 复杂 AOP 场景下的循环依赖:如果 Bean 存在循环依赖,并且被 AOP 代理,也无法解决;

3. 解决循环依赖

Spring 采用三级缓存解决循环依赖的问题;

一级缓存:用来保存完整的 Bean 对象;

二级缓存:用来保存半成品的对象;

三级缓存:用来保存 Bean 的工厂类;

假设 A 和 B 存在循环依赖:

    1. A 实例化,在属性注入的时候发现需要 B 对象,于是开始实例化 B,并将 A 保存到三级缓存;
    1. B 实例化,在属性注入的时候发现需要 A 对象,于是将 B 的工厂类也放到三级缓存;并调用三级缓存中 A 的工厂类,获取对象,并且把 Bean 放到二级缓存中,删除三级缓存中的工厂类;
    1. B 进行属性注入,从二级缓存中获取 A 对象,初始化完成之后,将 B 对象放到一级缓存;并从二级和三级缓存中删除;
    1. A 进行属性注入,并将对象放到一级缓存,并从二级缓存中删除;

实际上两级缓存就可以解决循环依赖的问题,二级缓存用于保存提前曝光的不完整的 Bean 对象,一级缓存用于保存完整的 Bean 对象;

Spring 使用三级缓存;目的是解决代理相关的循环依赖,使用三级缓存,代码的扩展性更好;

java 复制代码
public class DefaultSingletonBeanRegistry extends SimpleAliasRegistry implements SingletonBeanRegistry {

    ......

	/** Cache of singleton objects: bean name to bean instance. */
	private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);

	/** Creation-time registry of singleton factories: bean name to ObjectFactory. */
	private final Map<String, ObjectFactory<?>> singletonFactories = new ConcurrentHashMap<>(16);

	/** Cache of early singleton objects: bean name to bean instance. */
	private final Map<String, Object> earlySingletonObjects = new ConcurrentHashMap<>(16);
}

singletonObjects:一级缓存;

singletonFactories:三级缓存;

earlySingletonObjects:二级缓存;

二、SpringMVC 的流程

SpringMVC的请求响应步骤如下:

具体步骤:

1:(发起)发起请求到前端控制器(DispatcherServlet)

2:(查找)前端控制器请求查找 Handler(可以根据xml配置、注解进行查找)

3:(执行)请求适配器执行处理,并像前端返回 ModelAndView;

4.:(解析)前端进行视图解析;

5:(渲染)前端控制器进行视图渲染;

6:(响应)前端控制器向用户返回结果;


相关推荐
怒放吧德德4 小时前
Netty 4.2 入门指南:从概念到第一个程序
java·后端·netty
雨中飘荡的记忆6 小时前
大流量下库存扣减的数据库瓶颈:Redis分片缓存解决方案
java·redis·后端
心之语歌8 小时前
基于注解+拦截器的API动态路由实现方案
java·后端
华仔啊9 小时前
Stream 代码越写越难看?JDFrame 让 Java 逻辑回归优雅
java·后端
ray_liang10 小时前
用六边形架构与整洁架构对比是伪命题?
java·架构
用户83071968408210 小时前
spring ai alibaba + nacos +mcp 实现mcp服务负载均衡调用实战
spring boot·spring·mcp
Ray Liang11 小时前
用六边形架构与整洁架构对比是伪命题?
java·python·c#·架构设计
Java水解11 小时前
Java 中间件:Dubbo 服务降级(Mock 机制)
java·后端
SimonKing15 小时前
OpenCode AI辅助编程,不一样的编程思路,不写一行代码
java·后端·程序员
FastBean15 小时前
Jackson View Extension Spring Boot Starter
java·后端