@Configuration 注解的 Full 模式和 Lite 模式!

news2024/11/24 22:29:35

@Configuration 注解相信各位小伙伴经常会用到,但是大家知道吗,这个注解有两种不同的模式,一种叫做 Full 模式,另外一种则叫做 Lite 模式。

准确来说,Full 模式和 Lite 模式其实 Spring 容器在处理 Bean 时的两种不同行为。

这两种不同的模式在使用时候的表现完全不同,今天松哥就来和各位小伙伴捋一捋这两种模式。

1. 概念梳理

首先我们先来看一下 Spring 官方文档中对 Full 模式和 Lite 模式的一个介绍:

截图来自:https://docs.spring.io/spring-framework/reference/core/beans/java/basic-concepts.html

这个文档主要讲了这样几件事情:

  1. 我们可以通过在一个方法上添加 @Bean 注解,进而将该方法的返回值暴露给 Spring 容器,在这种场景下,@Bean 注解实际上就是一种通用的工厂方法机制。
  2. 当一个添加了 @Bean 注解的方法位于一个没有添加 @Configuration 注解的类里边时,那么这个添加了 @Bean 注解的方法在处理时就会按照 Lite 模式来处理。
  3. 当一个 Bean 被声明在添加了 @Component 注解的类中,那么会按照 Lite 模式来处理。
  4. 当一个 Bean 被声明在一个普通的类中时(plain old class),按照 Lite 模式来处理(这一点感觉和第二点差不多)。
  5. 在 Lite 模式下,@Bean 注解标记的方法最终不会被 CGLIB 进行代理,就是一个普通的工厂方法,因此,在 @Bean 标记的方法中,不能调用其他 @Bean 注解标记的方法,如果有需要,可以通过方法参数注入自己所需要的 Bean。
  6. 由于 Lite 模式下并不会使用 CGLIB,因此 @Bean 标记的方法可以是 final 类型的。
  7. 在大多数场景下,我们在一个 @Configuration 注解标记的类中,使用 @Bean 注解向 Spring 容器注册一个 Bean,都是 Full 模式。

官网文档的介绍还是有些抽象,接下来松哥通过具体的案例来和大家演示 Full 模式和 Lite 模式的差别。

2. Full 模式

先看 Full 模式,中文也可以称之为 完整 模式,我们平时使用时,在一个配置类上添加 @Configuration 注解,且不添加任何额外属性,这就是 Full 模式了。

Full 模式最大的特点是会给配置类通过 CGLIB 生成一个代理,所有被 @Bean 注解标记的方法将来都是通过代理方法进行调用。

假设我有如下配置类:

@Configuration
public class JavaConfig {

    @Bean
    User user() {
        return new User();
    }
}

现在,我们去 Spring 容器获取这个配置类:

public class JavaDemo {
    public static void main(String[] args) {
        AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(JavaConfig.class);
        JavaConfig config = ctx.getBean(JavaConfig.class);
        System.out.println("config.getClass() = " + config.getClass());
    }
}

打印结果如下:

大家看到,最终从 Spring 容器中拿到的 JavaConfig 实例并不是原始的 JavaConfig 对象,而是一个被代理的 JavaConfig 对象。

为什么要代理呢?肯定是为了实现某些功能。

大家看下面这个案例:

@Configuration
public class JavaConfig {

    @Bean
    User user() {
        User user = new User();
        user.setDog(dog());
        return user;
    }
    
    @Bean
    Dog dog() {
        return new Dog();
    }
}

在 Full 模式下,在 user() 方法中调用 dog() 方法的时候,调用的是一个代理对象的 dog 方法,在这个代理对象的 dog 方法中,会首先去检查 Spring 容器中是否存在 Dog 对象,如果存在,则直接使用 Spring 容器中的 dog 对象,就不会真正去执行 dog 方法而获取到一个新的 dog 对象了,如果 Spring 容器中不存在 dog 对象,才会创建新的 dog 对象出来。

一言以蔽之,在 Full 模式下,user 中的 dog 对象和 dog 方法注册到 Spring 容器的 dog 对象是同一个。

在 Full 模式下,由于要给当前类生成代理,然后去代理 @Bean 注解标记的方法,因此,这些 @Bean 注解标记的方法不能是 final 或者 private 类型的,因为 final 或者 private 类型的方法无法被重写,也就没法生成代理对象,如果添加了 final 或者 private 修饰符,那么会抛出如下异常:

3. Lite 模式

再来看 Lite 模式,这种模式可以认为是一种精简模式。

怎么开启呢?我们可以去除配置类上的 @Configuration 注解,或者去除之后添加 @Component 注解,又或者使用 @ComponentScan、@ImportResource、@Import 等注解标记类,那么最终都是 Lite 模式:

@Component
public class JavaConfig {

    @Bean
    final User user() {
        User user = new User();
        user.setDog(dog());
        return user;
    }

    @Bean
    Dog dog() {
        return new Dog();
    }
}

此时就是 Lite 模式,现在我们去 Spring 容器中获取这个配置类:

public class JavaDemo {
    public static void main(String[] args) {
        AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(JavaConfig.class);
        JavaConfig config = ctx.getBean(JavaConfig.class);
        System.out.println("config.getClass() = " + config.getClass());
    }
}

最终打印结果如下:

大家看到,我们从 Spring 容器中拿到的就是原始的对象,而不是一个被代理过的对象。因此:

  1. 由于 @Bean 注解标记的方法没有被代理,因此,该方法可以是 final 也可以是 private,运行时都不会报错。
  2. 由于 @Bean 方法没有被代理,因此在 user 方法中调用 dog 方法的时候,就直接调用了,这就导致 user 中的 dog 和最终 dog 方法注册到 Spring 容器中的 dog 不是同一个。

针对第二点,如果想要确保 user 中的 dog 和 Spring 容器中的 dog 是同一个,那么可以通过参数将所需要的对象注入进来,类似下面这样:

@Component
public class JavaConfig {

    @Bean
    final User user(Dog dog) {
        User user = new User();
        user.setDog(dog);
        return user;
    }

    @Bean
    Dog dog() {
        return new Dog();
    }
}

当 Spring 容器调用 user 方法初始化 User 对象时,发现该方法还有参数,因此会去容器中查找这个参数,找到了直接使用。

另外,我们也可以在类上添加 @Configuration 注解,但是通过修改属性值来启用 Lite 模式:

@Configuration(proxyBeanMethods = false)
public class JavaConfig {

    @Bean
    final User user(Dog dog) {
        User user = new User();
        user.setDog(dog);
        return user;
    }

    @Bean
    Dog dog() {
        return new Dog();
    }
}

如果设置了 proxyBeanMethods 属性为 false,那么也就是 Lite 模式了,其实我们从属性名称上也能看出来端倪:是否代理 @Bean 注解标记的方法。

4. 小结

总结一下:

  1. Lite 模式下,配置类中的方法就是普通方法,可以是 final 类型,也可以是 private。
  2. Lite 模式下,不需要通过 CGLIB 生成动态代理类,所以启动速度会快一些。
  3. Lite 模式下,一个 @Bean 方法调用另外一个 @Bean 方法,会导致同一个 Bean 被初始化两次。
  4. Full 模式下,会给配置类生成一个动态代理类,配置类中的所有方法都将被动态代理,因此配置类中的方法不能是 final 或者 private 的。
  5. Full 模式下,一个 @Bean 方法调用另外一个 @Bean 方法,动态代理方法会先去容器中检查是否存在该 Bean,如果存在,则直接使用容器中的 Bean,否则才会去创建新的对象。

日常开发中,我们使用较多的是 Full 模式。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/941661.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Nacos基础(2)——nacos的服务器和命名空间 springBoot整合nacos 多个nacos配置的情况

目录 引出nacos服务器和命名空间Nacos服务器命名空间 springBoot整合nacosspringcloud Alibaba 版本与springcloud对应关系引包配置maincontroller 报错以及解决【报错】错误:缺少服务名称报错:9848端口未开放 启动测试引入多个nacos配置多个配置的情况没…

基于java swing和mysql实现的学生选课成绩信息管理系统(源码+数据库+ER图文档+运行指导视频)

一、项目简介 本项目是一套基于java swing和mysql实现的学生选课成绩信息管理系统,主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的Java学习者。 包含:项目源码、项目文档、数据库脚本等,该项目附带全部源码可作为毕设使用。…

概念解析 | 无人机集群形状与轨迹建模: 集群舞蹈的艺术

注1:本文系“概念解析”系列之一,致力于简洁清晰地解释、辨析复杂而专业的概念。本次辨析的概念是:无人机集群形状和轨迹建模 无人机集群形状与轨迹建模: 集群舞蹈的艺术 无人机集群的形状和轨迹建模可能听起来像是一部科幻小说的标题,但它实际上是现实中的一个重要研究领…

国内精选五大现货黄金价格实时查询软件最新名单(综合榜单)

随着金融市场的不断发展和投资者的日益增多,现货黄金价格实时查询软件成为了人们关注的焦点。投资者需要一款功能强大、操作简便、数据准确的软件来帮助他们获取实时的黄金价格信息,以便做出更加明智的投资决策。 本文将介绍国内精选五大现货黄金价格实…

iOS - 订阅型内购指南

一、App Store Connect 帮助 二、测试 三、订阅状态 四、问题思考 1、订阅归属: 以往的消耗性内购, 通常会生成订单ID对应到苹果的内购ID及用户id,对于我们来说,内购仅仅只是个支付工具,而订阅型内购有一整套销售模型订阅内购…

zabbix语言无法选择中文--zabbix安装配置中文

You are not able to choose some of the languages, because locales for them are not installed on the web server. 1、安装wget yum -y install wget 2、下载中文中文字体并配置 wget https://github.com/echohn/zabbix-zh_CN/archive/master.zip yum -y install unzip un…

4年经验来面试20K的测试岗,一问三不知,我还真不如去招应届生...

公司前段缺人,也面了不少测试,结果竟然没有一个合适的。一开始瞄准的就是中级的水准,也没指望来大牛,提供的薪资在10-20k,面试的人很多,但平均水平很让人失望。看简历很多都是4年工作经验,但面试…

恒运资本:A股三大指数是什么?A股三大指数怎么看?

炒股并不是盲目跟风,投资者自身要了解股票基本常识。例如指数反映的是股票商场上股票的变动状况,能够给我们的投资决策带来一定的依据。那么,A股三大指数是什么?A股三大指数怎么看?恒运资本为我们准备了相关内容&#…

如何给图片加水印?

如何给图片加水印?在我们的日常生活中,许多热爱摄影的朋友都会选择给自己的照片添加水印。这是因为我们深知,一张出色的照片背后需要付出大量的努力和心血,而通过添加水印可以有效地保护自己照片的版权,这样即使将图片…

MySQL概述,架构原理

一.MySQL简介 MySQL是一个关系型数据库管理系统,由瑞典的MySQL AB公司开发,后被oracle公司收购,MySQL是当下最流行的关系型数据库管理系统之一,在WEB应用方面,MySQL是最好的RDBMS(Relational Database Man…

得帆信息西区总经理——何龙:低代码初识

企业数字化建设、数字化转型是近年来企业经营管理必然面对的课题;相较于面对传统经营管理的驾车就熟,这无疑给企业管理者提出了新的课题和新的挑战。在当前新技术新生产力不断涌现、新市场特点不断变化的时代,企业在要练好内功、加强经营管理…

龙讯旷腾Q-Studio新增力场优化功能

Q-Studio新功能 Q-Studio(在线建模功能)依托Mcloud平台免费向用户开放使用,基于jsmol的建模功能无需安装任何软件/插件,通过web端即可在线完成格式转换和可视化建模工作,并可对模型进行个性化的二次编辑,快…

【80天学习完《深入理解计算机系统》】第九天 3.2 数据传送指令【mov】【栈和堆 就是内存】【leaq】【一元操作】【二元操作】

专注 效率 记忆 预习 笔记 复习 做题 欢迎观看我的博客,如有问题交流,欢迎评论区留言,一定尽快回复!(大家可以去看我的专栏,是所有文章的目录)   文章字体风格: 红色文字表示&#…

开源利器推荐:美团动态线程池框架的接入分享及效果展示

前言 蛮早前有些过关于线程池的使用及参数的一些参考配置,有兴趣的可以翻看以前的博文,但终究无法解决线程池的动态监控和实时修改。 以前读过美团早期发布的动态线程池框架的思路相关文章,但想要独自实现不是一件容易的事。 去年&#xff0c…

SpringMVC之RESTful(含实际代码操作)

文章目录 前言一、RESTful简介二、RESTful的实现三、HiddenHttpMethodFilter四、源码实例1.工程文件图2.Employee实体类2.dao层:EmployeeDao类3.首页:index.html4.控制层:EmployeeController类5.employee_add.html页面6.employee_list.html页…

阀门状态监测和预测性维护的原理和实施步骤

随着制造业数字化转型的推进,预测性维护(Predictive Maintenance,简称PdM)成为提高生产效率和设备可靠性的关键策略之一。在流程工厂中,阀门作为重要的设备之一,起着控制流体流动的关键作用。本文将探讨如何…

基于AVR128单片机智能传送装置

一、系统方案 1、板载可变电阻(电位器)R29的电压作为处理器ATmega128的模数转换模块中单端ADC0的模拟信号输入(跳线JP13短接)。 2、调节电位器,将改变AD转换接口ADC0的模拟信号输入,由处理器完成ADC0的A/D转…

积分商城系统源码_积分兑换礼品功能设计_OctShop

礼品系统:礼品需要积分兑换,买家在平台消费,评价商品或店铺赚取积分,商家发布礼品可赚取积分,商家积分可用于奖励买家评价商品。礼品系统和商品库差不多,礼品的发布,修改,展示,库存以…

vue中bus的使用和涉及到的问题

创建一个js文件 import Vue from "Vue" export default new Vue 我们可以直接在要使用的页面中引用使用 import bus from /assets/js/eventBus.js;bus.$emit("info", "123") // 使用bus.$on("info", (val) > { // 接收console.l…

项目透明度如何改善团队的工作流程?

无论项目简单还是复杂,项目透明度一直是项目过程中的重要因素。在当今快节奏的商业环境中,对透明度的关注与日俱增,现已成为团队及其项目成功的关键因素。 但创建一个透明的流程对团队管理而言,是一个重大挑战,因团队…