当前位置: 首页 > news >正文

@Lazy原理与实战

文章目录

  • 前言
  • 一、@Lazy的解析
    • 1.1、场景一
    • 1.2、场景二
  • 二、@Lazy Bean的处理
  • 三、解决循环依赖的问题


前言

  @Lazy是Spring提供的注解,用于实现懒加载功能。该注解既可以修饰方法,也可以修饰类,字段等:

  • @Lazy修饰的类,不会在Spring容器的refresh方法中进行实例化,而是只有在第一次注入或使用该 Bean 时才会实例化。
@Lazy
@Component
public class MyService {// ...
}
  • @Lazy修饰的方法,该方法创建的 Bean 是懒加载的,只有第一次被注入或调用时才会初始化。
@Configuration
public class AppConfig {@Bean@Lazypublic ExpensiveBean expensiveBean() {return new ExpensiveBean();}
}
  • @Lazy修饰的字段,会延迟注入。
@Component
public class SomeComponent {@Autowired@Lazyprivate AnotherService anotherService;
}
  • @Lazy修饰的构造函数参数或方法参数,参数所代表的 Bean 会被延迟注入。
@Component
public class MyComponent {private final HeavyService heavyService;public MyComponent(@Lazy HeavyService heavyService) {this.heavyService = heavyService;}
}

一、@Lazy的解析

  @Lazy的解析时机,有两个场景:

场景处理逻辑
@ComponentScan 自动扫描组件ClassPathBeanDefinitionScannerprocessCommonDefinitionAnnotations() 中设置 lazyInit
手动注册配置类(如 registerBean(SomeClass.class)调用 processCommonDefinitionAnnotations() 设置 lazyInit

1.1、场景一

  对应的案例工程:

public class Demo1 {public static void main(String[] args) {// 初始化一个空容器AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();// 手动注册类 A,而不是使用 @ComponentScanAnnotatedBeanDefinitionReader reader = new AnnotatedBeanDefinitionReader(context);reader.register(A.class);// 刷新容器,触发 Bean 生命周期context.refresh();System.out.println("IOC 容器已启动");// 懒加载:此时 A 还未被创建System.out.println("开始获取 A");A a = context.getBean(A.class);System.out.println("A 获取完成: " + a);}
}@Lazy
@Component
class A {public A() {System.out.println("A 被创建");}
}

  在AnnotationConfigUtils#processCommonDefinitionAnnotations时调用。
在这里插入图片描述

1.2、场景二

  对应的案例工程:

public class Demo2 {public static void main(String[] args) {AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(Config.class);System.out.println("IOC容器已启动");// 不会实例化 A,只有调用 getBean 时才会触发创建A a = context.getBean(A.class);System.out.println("B 已获取: " + a);}
}@Configuration
@ComponentScan
class Config {
}@Lazy
@Component
class B {public B() {System.out.println("B 初始化");}
}

  对应的解析时机是,refresh方法中,调用所有bean工厂的后置处理器,ConfigurationClassPostProcessor扫描并解析@ComponentScan注解的逻辑中:在这里插入图片描述
  parse最终会进入doScan方法:
在这里插入图片描述
  同样是在processCommonDefinitionAnnotations中设置bean定义的lazy属性:
在这里插入图片描述
  无论上述哪一个场景,最终的目的都是为了将bean定义的lazyInit属性进行设置,用于后续的判断。

二、@Lazy Bean的处理

  对于标注了@Lazy的Bean的处理,主要体现在两个地方:

  • 初始化所有非懒加载的单例bean
  • bean生命周期的属性注入

  在初始化所有非懒加载的单例bean时,如果发现某个bean定义的lazyInit属性为true,就不会对该bean进行实例化。
在这里插入图片描述
  而在bean生命周期的依赖注入阶段,当发现了某个注入点的bean定义的lazyInit属性为true:
在这里插入图片描述
  则会在buildLazyResolutionProxy内部直接生成一个代理对象并注入给对应的字段:
在这里插入图片描述
  生成代理对象。
在这里插入图片描述
  在实际调用时,执行TargetSource的回调,实际执行的也是DefaultListableBeanFactorydoResolveDependency方法。
在这里插入图片描述

三、解决循环依赖的问题

  使用@Lazy注解,也可以解决循环依赖的问题:

public class Demo3 {public static void main(String[] args) {AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(Config1.class);}
}@Configuration
@ComponentScan
class Config1{}@Component
class C{@Lazy@Autowiredprivate D d;}@Component
class D{@Lazy@Autowiredprivate C c;}

  在上面的场景中,当C实例化完成,依赖注入D时,发现D是加上了@Lazy注解,就不会去Spring容器中去找D,走找不到创建的流程,而是直接生成一个D的代理对象,注入给C的D属性,然后C执行初始化操作,也就打破了循环依赖。
  当C中的D属性被调用时,才会回调切面的TargetSource,走DefaultListableBeanFactorydoResolveDependency方法,从Spring的容器中获取bean,获取不到就去创建,走doCreateBean的流程。(注意,同样无法解决两者都是构造方法的循环依赖问题,但是可以解决原型bean的循环依赖)

相关文章:

  • vivado IP核High speed/Low latency设置对系统性能的影响
  • ELK实现nginx、mysql、http的日志可视化实验
  • 搭建仿真yolo环境
  • 第三章 k近邻法
  • Netty自定义协议解析
  • 【MATLAB代码】基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),附源代码|订阅专栏后可直接查看
  • mac 安装homebrew (nvm 及git)
  • [25-cv-06246]Keith律所代理黑蝴蝶版权画
  • WEB3全栈开发——面试专业技能点P8DevOps / 区块链部署
  • 一个有用的geotools上传sld样式与shp文件以生成缩略图的方法
  • ShaderToy:光晕效果
  • Conda安装pytorch和cuda出现问题的解决记录
  • 【投稿优惠】2025年航天技术 、雷达信号与无人机应用国际会议 (ATRA 2025)
  • 组件库实战-基建思路
  • 缓存一致性性的 实现等价
  • 免费好用的专业提词器有哪些~~~
  • Python训练营-Day26-函数专题1:函数定义与参数
  • 我的世界进阶模组开发教程——渲染学(3)
  • 安宝特方案丨从依赖经验到数据驱动:AR套件重构特种装备装配与质检全流程
  • 安宝特方案丨船舶智造的“AR+AI+作业标准化管理解决方案”(装配)
  • 保定网站制作套餐/腾讯企点app下载安装
  • 商城类网站建设需要多少钱/aso优化app推广
  • 做英语陪同翻译兼职的网站/什么是整合营销并举例说明
  • 企业如何选择网站/百度的官方网站
  • 如何找人帮我做网站推广/谷歌官方网站
  • 做商务网站/网站设计案例