深入理解 Spring 的 Lazy Loading:原理、实现与应用场景
延迟加载(Lazy Loading)是 Spring 容器管理 Bean 的一种策略,指 只有在需要时(调用 getBean() 方法获取 Bean 时)才会实例化该 Bean。这是 Spring 提供的一种优化机制,用于提高启动效率和降低资源占用。
1. 延迟加载的含义
- 在延迟加载模式下,Spring 容器初始化时不会立即实例化所有 Bean,而是等到真正需要使用时(即调用
getBean()方法时),才创建 Bean 实例。 - 如果不启用延迟加载(非 Lazy 模式),则所有单例(
singleton) Bean 会在容器启动时立即实例化。
2. 延迟加载的优点
-
节省资源:
- 容器启动时不需要加载和创建所有 Bean,启动速度更快。
- 避免不必要的对象实例化,降低内存占用,尤其是对于未使用的 Bean。
-
适合轻量级应用:
- 在资源有限的环境中(如嵌入式系统),延迟加载可以显著优化性能。
- 在开发或测试阶段,也可以通过延迟加载缩短启动时间。
-
按需加载:
- 只有在确实需要使用某个 Bean 时,才会创建它的实例,避免初始化不必要的逻辑。
3. 延迟加载的实现方式
(1) 使用 @Lazy 注解
在类或方法上添加 @Lazy 注解,可以让单例 Bean 延迟加载。
示例代码:
@Component
@Lazy // 延迟加载
public class ExpensiveBean {public ExpensiveBean() {System.out.println("ExpensiveBean created!");}
}
测试代码:
ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
System.out.println("Container initialized.");
ExpensiveBean bean = context.getBean(ExpensiveBean.class); // 此时 Bean 才会创建
运行结果:
Container initialized.
ExpensiveBean created!
解释:
@Lazy生效后,ExpensiveBean不会在容器启动时被创建。- 只有在调用
getBean()时,ExpensiveBean才会被实例化。
(2) 如果未使用 @Lazy
代码:
@Configuration
public class AppConfig {@Beanpublic ExpensiveBean expensiveBean() {return new ExpensiveBean(); // 无 @Lazy,默认预加载}
}
运行结果:
ExpensiveBean created!
Container initialized.
解释:
- 默认情况下,
ApplicationContext会在启动时实例化所有单例 Bean,所以在输出"Container initialized."前就会创建ExpensiveBean。
(3) XML 配置方式
在 XML 文件中为 Bean 配置延迟加载:
<bean id="expensiveBean" class="com.example.ExpensiveBean" lazy-init="true"/>
(4) 在 @Configuration 中使用
对于使用 Java 配置的项目,可以在配置类中的 @Bean 方法上添加 @Lazy:
@Configuration
public class AppConfig {@Bean@Lazypublic ExpensiveBean expensiveBean() {return new ExpensiveBean();}
}
4. 延迟加载的默认行为
(1) BeanFactory 的默认行为
BeanFactory是一个轻量级容器,默认使用延迟加载策略。- 它不会在容器启动时实例化任何 Bean,而是等到调用
getBean()方法时才实例化。
(2) ApplicationContext 的默认行为
ApplicationContext是 Spring 容器的常用实现,默认会在启动时预加载所有单例(singleton) Bean。- 例外情况:
- 如果使用
@Lazy注解或 XML 中的lazy-init="true",单例 Bean 会延迟加载。 - 非单例(
prototypeScope)的 Bean 本身默认是延迟加载的。
- 如果使用
5. 延迟加载与非延迟加载的对比
| 行为 | 延迟加载 | 非延迟加载 |
|---|---|---|
| 实例化时机 | 调用 getBean() 时实例化 | 容器启动时实例化 |
| 适用范围 | 需要优化启动时间或资源消耗的场景 | 高性能服务器或需要预加载依赖的场景 |
| 容器类型 | BeanFactory 默认延迟加载 | ApplicationContext 默认非延迟加载 |
| 预加载的 Bean | 无(除非显式调用) | 单例 Bean 默认全部实例化 |
| 性能表现 | 启动速度快,可能会导致首次调用延迟 | 启动时占用更多内存,运行时性能更高 |
6. 示例场景
-
资源密集型对象的加载:
- 如果某个 Bean 的创建非常耗时(例如连接外部服务),使用延迟加载可以避免容器启动时的性能瓶颈。
-
开发与测试阶段:
- 在开发和测试中,延迟加载可以缩短容器的启动时间,提升开发效率。
-
条件性加载:
- 某些 Bean 只有在特定条件下才会被使用,延迟加载可以避免不必要的资源消耗。
7. 总结
- 延迟加载的本质: Bean 只有在第一次使用时才会被实例化。
- 优点: 减少容器启动时的资源消耗,适合轻量级或资源受限的场景。
- 默认行为:
BeanFactory默认延迟加载,而ApplicationContext默认预加载单例 Bean。 - 实现方式: 使用
@Lazy注解或 XML 配置轻松启用延迟加载。
通过灵活地使用延迟加载,可以显著优化应用程序的启动时间和资源利用率,特别是在复杂项目或资源密集型应用中。
相关文章:
深入理解 Spring 的 Lazy Loading:原理、实现与应用场景
延迟加载(Lazy Loading)是 Spring 容器管理 Bean 的一种策略,指 只有在需要时(调用 getBean() 方法获取 Bean 时)才会实例化该 Bean。这是 Spring 提供的一种优化机制,用于提高启动效率和降低资源占用。 1.…...
扬帆数据结构算法之雅舟航程,漫步C++幽谷——LeetCode刷题之移除链表元素、反转链表、找中间节点、合并有序链表、链表的回文结构
人无完人,持之以恒,方能见真我!!! 共同进步!! 文章目录 一、移除链表元素思路一思路二 二、合并两个有序链表思路:优化: 三、反转链表思路一思路二 四、链表的中间节点思…...
【unity游戏开发之InputSystem——02】InputAction的使用介绍(基于unity6开发介绍)
文章目录 一、InputAction简介1、InputAction是什么?2、示例 二、InputAction参数相关1、点击齿轮1.1 Actions 动作(1)动作类型(Action Type)(2)初始状态检查(Initial State Check&a…...
Excel常用功能总结
Excel 是微软办公软件套装中的一个重要组件,用于数据处理和分析。以下是一些 Excel 的常用功能总结: 基本操作 1.单元格操作:选择、插入、删除单元格、行或列。 2.数据输入:输入文本、数字、日期和时间。 3.格式设置:设…...
【go语言】变量和常量
一、变量 1.1 变量的定义 程序 : 我们向电脑说了一段话,需要电脑才能理解 (沟通机制 ,xxx语言 -- 汇编 -- 机器码),电脑实际上识别的是机器码 : 0 1 1 1 0 1 (高低电频)…...
Node.js——express中间件(全局中间件、路由中间件、静态资源中间件)
个人简介 👀个人主页: 前端杂货铺 🙋♂️学习方向: 主攻前端方向,正逐渐往全干发展 📃个人状态: 研发工程师,现效力于中国工业软件事业 🚀人生格言: 积跬步…...
大语言模型的语境中“越狱”和思维链
大语言模型的语境中“越狱”和思维链 越狱(Jailbreaking) 含义:在大语言模型的语境中,“越狱”是指用户试图绕过语言模型的安全限制和使用规则,让模型生成违反伦理道德、包含有害内容(如暴力、歧视、恶意软件代码等)的输出。这些安全限制是由模型开发者设置的,目的是确…...
JAVA学习记录4
文章为个人学习记录,仅供参考,如有错误请指出。 上期说到IDEA的安装,具体的使用方法就不记录了。这篇主要记录一些基础语法。 类型转换-自动类型转换 类型范围小的变量,可以直接赋值给类型范围大的变量。 在表达式中&…...
手机网络性能测试仪器介绍
手机网络性能测试仪器是用于检测和评估手机网络性能的精密设备。这些仪器通常具备多种测试功能,以确保手机在不同网络环境下的表现都能得到准确评估。以下是对手机网络性能测试仪器的详细介绍: 一、主要类型 手机综合测试仪:如R&SCMU200…...
vue3+ts watch 整理
watch() 一共可以接受三个参数,侦听数据源、回调函数和配置选项 作用:监视数据的变化(和Vue2中的watch作用一致) 特点:Vue3中的watch只能监视以下四种数据: ref定义的数据。 reactive定义的数据。 函数返…...
【Elasticsearch入门到落地】6、索引库的操作
接上篇《5、安装IK分词器》 上一篇我们进行了IK分词器的安装与测试,本篇我们来学习ElasticSearch的索引库的操作,学习mapping映射属性以及CRUD操作。 一、前情回顾 我们在前几篇学习了ElasticSearch的基本概念,并动手搭建了ElasticSearch环…...
Java TCP可靠传输(1)
TCP 可靠传输 一. 确认应答 由发送方填充,再由接收方在序号的基础上1,填充到确认序号中,来表示已经接收到前面发送的,表明下一个从哪个位置发送。 二. 超时重传 数据在网络上传输时会经过很多网络设备,如果其中一个…...
ipad和macbook同步zotero文献附件失败的解决办法
背景:我所有的文献及其附件pdf都是在台式机(windows系统),想要把这些文献同步到云上,然后再从云上同步到平板和其他笔记本电脑比如macbook。文献同步虽已成功,但文献附件都无法打开。 平板报错如下…...
linux-ubuntu学习笔记碎记
~指/home/user_name这个目录 查看软件安装目录:whereis vim 查看当前路径:pwd 终端中键入ctrls会挂起终端,即终端不响应键鼠;ctrlq可以恢复。 和虚拟机开启共享文件夹互传文件 点击桌面,按ctrlaltt,开…...
RV1126+FFMPEG推流项目(11)编码音视频数据 + FFMPEG时间戳处理
本节介绍本章节主要讲解的是push_server_thread线程的具体处理流程, push_server_thread这个线程的主要功能是通过时间戳比较,来处理音频、视频的数据并最终推流到SRT、RTMP、UDP、RTSP服务器 push_server_thread:流程如下 上图,…...
人工智能的出现,给生命科学领域的研究带来全新的视角|行业前沿·25-01-22
小罗碎碎念 今天和大家分享一份白皮书,系统总结并陈述人工智能在生命科学领域的应用。 人工智能在生命科学领域的应用,具体包括——单细胞转录组、疾病诊疗、医疗文本处理、RNA结构预测等多个方面,通过这份报告,我们可以详细了解相…...
python注释格式总结
三个双引号的用于文件,类,函数注释。 没有统一的规定,以下是比较清晰的写法。 文件注释(文件顶部):文件用途空行作者信息(每行一个键:值) 类注释(类名下行)…...
Django实现数据库的表间三种关系
Django实现数据库的表间三种关系 1. 一对多(One-to-Many)关系示例:关系说明:查询示例: 2. 一对一(One-to-One)关系示例:关系说明:查询示例: 3. 多对多&#x…...
C++蓝桥真题讲解
本篇文章和大家一起来试试一些简单的蓝桥真题 注意:本篇文章将全程使用devc和蓝桥官网,如果有小伙伴找不到devc安装包的可以本篇文章中下载。 赛前必知点 1.正式比赛时,先从蓝桥官网下载题目文档,然后用devc进行编译,…...
【21】Word:德国旅游业务❗
目录 题目 NO1.2.3 NO4 NO5.6 NO7 NO8.9.10.11 题目 NO1.2.3 F12:另存为布局→页面设置→页边距:上下左右选中“德国主要城市”→开始→字体对话框→字体/字号→文本效果:段落对话框→对齐方式/字符间距/段落间距 NO4 布局→表对话框…...
vscode里如何用git
打开vs终端执行如下: 1 初始化 Git 仓库(如果尚未初始化) git init 2 添加文件到 Git 仓库 git add . 3 使用 git commit 命令来提交你的更改。确保在提交时加上一个有用的消息。 git commit -m "备注信息" 4 …...
从WWDC看苹果产品发展的规律
WWDC 是苹果公司一年一度面向全球开发者的盛会,其主题演讲展现了苹果在产品设计、技术路线、用户体验和生态系统构建上的核心理念与演进脉络。我们借助 ChatGPT Deep Research 工具,对过去十年 WWDC 主题演讲内容进行了系统化分析,形成了这份…...
3.3.1_1 检错编码(奇偶校验码)
从这节课开始,我们会探讨数据链路层的差错控制功能,差错控制功能的主要目标是要发现并且解决一个帧内部的位错误,我们需要使用特殊的编码技术去发现帧内部的位错误,当我们发现位错误之后,通常来说有两种解决方案。第一…...
vscode(仍待补充)
写于2025 6.9 主包将加入vscode这个更权威的圈子 vscode的基本使用 侧边栏 vscode还能连接ssh? debug时使用的launch文件 1.task.json {"tasks": [{"type": "cppbuild","label": "C/C: gcc.exe 生成活动文件"…...
Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务
通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...
全球首个30米分辨率湿地数据集(2000—2022)
数据简介 今天我们分享的数据是全球30米分辨率湿地数据集,包含8种湿地亚类,该数据以0.5X0.5的瓦片存储,我们整理了所有属于中国的瓦片名称与其对应省份,方便大家研究使用。 该数据集作为全球首个30米分辨率、覆盖2000–2022年时间…...
高防服务器能够抵御哪些网络攻击呢?
高防服务器作为一种有着高度防御能力的服务器,可以帮助网站应对分布式拒绝服务攻击,有效识别和清理一些恶意的网络流量,为用户提供安全且稳定的网络环境,那么,高防服务器一般都可以抵御哪些网络攻击呢?下面…...
ABAP设计模式之---“简单设计原则(Simple Design)”
“Simple Design”(简单设计)是软件开发中的一个重要理念,倡导以最简单的方式实现软件功能,以确保代码清晰易懂、易维护,并在项目需求变化时能够快速适应。 其核心目标是避免复杂和过度设计,遵循“让事情保…...
Linux C语言网络编程详细入门教程:如何一步步实现TCP服务端与客户端通信
文章目录 Linux C语言网络编程详细入门教程:如何一步步实现TCP服务端与客户端通信前言一、网络通信基础概念二、服务端与客户端的完整流程图解三、每一步的详细讲解和代码示例1. 创建Socket(服务端和客户端都要)2. 绑定本地地址和端口&#x…...
MySQL 8.0 事务全面讲解
以下是一个结合两次回答的 MySQL 8.0 事务全面讲解,涵盖了事务的核心概念、操作示例、失败回滚、隔离级别、事务性 DDL 和 XA 事务等内容,并修正了查看隔离级别的命令。 MySQL 8.0 事务全面讲解 一、事务的核心概念(ACID) 事务是…...
