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:(响应)前端控制器向用户返回结果;


相关推荐
CodeToGym17 小时前
【Java 办公自动化】Apache POI 入门:手把手教你实现 Excel 导入与导出
java·apache·excel
凡人叶枫17 小时前
C++中智能指针详解(Linux实战版)| 彻底解决内存泄漏,新手也能吃透
java·linux·c语言·开发语言·c++·嵌入式开发
JMchen12317 小时前
Android后台服务与网络保活:WorkManager的实战应用
android·java·网络·kotlin·php·android-studio
惊讶的猫17 小时前
Redis持久化介绍
数据库·redis·缓存
阔皮大师17 小时前
INote轻量文本编辑器
java·javascript·python·c#
小法师爱分享17 小时前
StickyNotes,简单便签超实用
java·python
qq_2975746718 小时前
Linux 服务器 Java 开发环境搭建保姆级教程
java·linux·服务器
金牌归来发现妻女流落街头18 小时前
【从SpringBoot到SpringCloud】
java·spring boot·spring cloud
毅炼18 小时前
Java 基础常见问题总结(4)
java·后端
张3蜂18 小时前
深入理解 Python 的 frozenset:为什么要有“不可变集合”?
前端·python·spring