@Lazy原理与实战

文章目录


前言

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

  • @Lazy修饰的类,不会在Spring容器的refresh方法中进行实例化,而是只有在第一次注入或使用该 Bean 时才会实例化。
java 复制代码
@Lazy
@Component
public class MyService {
    // ...
}
  • @Lazy修饰的方法,该方法创建的 Bean 是懒加载的,只有第一次被注入或调用时才会初始化。
java 复制代码
@Configuration
public class AppConfig {
    @Bean
    @Lazy
    public ExpensiveBean expensiveBean() {
        return new ExpensiveBean();
    }
}
  • @Lazy修饰的字段,会延迟注入。
java 复制代码
@Component
public class SomeComponent {
    @Autowired
    @Lazy
    private AnotherService anotherService;
}
  • @Lazy修饰的构造函数参数或方法参数,参数所代表的 Bean 会被延迟注入。
java 复制代码
@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、场景一

对应的案例工程:

java 复制代码
public class Demo1 {
    public static void main(String[] args) {
        // 初始化一个空容器
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();

        // 手动注册类 A,而不是使用 @ComponentScan
        AnnotatedBeanDefinitionReader 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、场景二

对应的案例工程:

java 复制代码
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注解,也可以解决循环依赖的问题:

java 复制代码
public class Demo3 {
    public static void main(String[] args) {
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(Config1.class);
    }
}


@Configuration
@ComponentScan
class Config1{
    
    
}

@Component
class C{
    
    @Lazy
    @Autowired
    private D d;

}


@Component
class D{
    
    @Lazy
    @Autowired
    private C c;

}

在上面的场景中,当C实例化完成,依赖注入D时,发现D是加上了@Lazy注解,就不会去Spring容器中去找D,走找不到创建的流程,而是直接生成一个D的代理对象,注入给C的D属性,然后C执行初始化操作,也就打破了循环依赖。

当C中的D属性被调用时,才会回调切面的TargetSource,走DefaultListableBeanFactorydoResolveDependency方法,从Spring的容器中获取bean,获取不到就去创建,走doCreateBean的流程。(注意,同样无法解决两者都是构造方法的循环依赖问题,但是可以解决原型bean的循环依赖)

相关推荐
一 乐4 小时前
校园墙|校园社区|基于Java+vue的校园墙小程序系统(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·后端·小程序
阿巴~阿巴~4 小时前
IPv4地址转换函数详解及C++容器安全删除操作指南
linux·服务器·c++·网络协议·算法·c++容器安全删除操作·ipv4地址转换函数
TT哇4 小时前
【面经 每日一题】面试题16.25.LRU缓存(medium)
java·算法·缓存·面试
青云交4 小时前
Java 大视界 -- 基于 Java 的大数据联邦学习在跨行业数据协同创新中的实践突破
java·分布式计算·隐私保护·apache flink·大数据联邦学习·跨行业数据协同·安全通信
合作小小程序员小小店4 小时前
桌面开发,在线%考试管理%系统,基于eclipse,java,swing,mysql数据库。
java·数据库·mysql·eclipse·jdk
oioihoii4 小时前
C/C++混合项目中的头文件管理:.h与.hpp的分工与协作
java·c语言·c++
一瓢一瓢的饮 alanchan4 小时前
Flink原理与实战(java版)#第2章 Flink的入门(第二节Flink简介)
java·大数据·flink·kafka·实时计算·离线计算·流批一体化计算
vx_bscxy3224 小时前
告别毕设焦虑!Python 爬虫 + Java 系统 + 数据大屏,含详细开发文档 基于微信小程序的民宿预约系统22398 (上万套实战教程,赠送源码)
java·spring boot·mysql·微信小程序·课程设计
应茶茶5 小时前
VsCode通过SSH远程连接云服务器遇到主机密钥变更问题
服务器·vscode·ssh
skywalk81635 小时前
FreeBSD 14.3 轻量级Jail虚拟机:内存资源占用仅13MB的实战指南
运维·服务器·freebsd·jail