精简还是全能?如何在 Full 和 Lite 之间做出最佳选择!关于Configuration注解的Full模式与Lite模式(SpringBoot2)

在这里插入图片描述

  • 🏃‍♂️ 微信公众号: 朕在debugger
  • © 版权: 本文由【朕在debugger】原创、需要转载请联系博主
  • 📕 如果文章对您有所帮助,欢迎关注、点赞、转发和订阅专栏!


前言

关于 @Configuration 注解,相信在座的各位 Javaer 都使用过,且大部分人使用它是直接在配置类上塞一个 @Configuration 就完事了,不会去过多使用它的参数,这期文章是来讲述 @Configuration 注解的一个代理相关的参数 proxyBeanMethod。

回顾

说是讲 proxyBeanMethod 参数,但是注解本身的功能也需要大致过一下,@Configuration 注解是 Spring 框架中的一个核心注解,用于定义配置类。

配置类可以包含定义和生成 SpringBean 的方法。这些方法通过 @Bean 注解进行标记。(默认是单实例的)

同时加上 @Configuration 注解的类也允许定义额外的bean( @Import )或导入其他配置类( @ImportResource )。


文章目录

  • 一、何为 Full 模式?何为 Lite 模式?查看源码辩雌雄
    • Full 模式
    • Lite 模式
  • 二、案例验证
    • 2-1、将 proxyBeanMethods 设置为 true(默认)
    • 2-2、将 proxyBeanMethods 设置为 false
  • 三、何时使用 Full 模式与 Lite 模式?
  • 四、总结


一、何为 Full 模式?何为 Lite 模式?查看源码辩雌雄

说 Full,说 Lite,其实都是在说 proxyBeanMethods 这个参数,也就是代理 bean 的方法。

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Configuration {
    @AliasFor(
        annotation = Component.class
    )
    String value() default "";

    boolean proxyBeanMethods() default true;

    boolean enforceUniqueMethods() default true;
}

可以看出,proxyBeanMethods 方法返回值是一个 Boolean 值,默认值是 true,什么意思呢?

它的值为 true 时,就是 Full 模式。为 false 时,就是 Lite 模式。

Full 模式

Full 模式是当你在类上使用 @Configuration 注解且没有将 proxyBeanMethods 属性显式设置为 false 时(默认为 true )的模式。

在 Full 模式下,Spring 容器会通过 CGLIB 动态代理技术创建配置类的代理实例。

这样做的目的是保证同一个配置类中对其他 @Bean 方法的调用总是返回相同的实例,即保证 @Bean 方法是以单例的方式被管理的,无论它们被调用多少次。

这对于维护配置类中 Bean 之间的依赖关系非常重要,因为它确保了依赖注入的一致性和 Spring 容器中 Bean 的单例特性。

Lite 模式

Lite 模式是当你在类上使用 @Component、@Service、@Repository 等其他组件注解,或者使用 @Configuration 注解但将 proxyBeanMethods 属性显式设置为 false 时的模式。

在 Lite 模式下,Spring 容器不会创建配置类的 CGLIB 代理。

这意味着,配置类中的 @Bean 方法在每次调用时都会生成一个新的实例(除非你自己在方法内部处理单例逻辑),并且这些方法之间的调用就像普通 Java 方法调用一样,不会被 Spring 容器特别处理。

Lite 模式通常用于轻量级的 Bean 定义,或者当配置类中的 @Bean 方法不需要相互依赖时,可以提高性能,减少资源消耗。

二、案例验证

定义一个配置类,定义了两个Bean:一个数据库连接池(DataSource)和一个 JdbcTemplate,后者依赖于前者。

2-1、将 proxyBeanMethods 设置为 true(默认)

@Configuration(proxyBeanMethods = true) // 默认为true,可以不显式设置
public class AppConfig {

    // 创建了一个DataSource实例
    @Bean
    public DataSource dataSource() {
        return new HikariDataSource();
    }

    @Bean
    public JdbcTemplate jdbcTemplate() {
        // 这里调用了dataSource()方法来获取DataSource实例
        return new JdbcTemplate(dataSource());
    }
}

在这个例子中,因为 proxyBeanMethods 设置为 true,当 Spring 调用 jdbcTemplate() 方法来创建 JdbcTemplate Bean 时,内部通过调用 dataSource() 方法得到的 DataSource 实例,将会是同一个实例。

这是因为配置类被 CGLIB 代理,确保了 dataSource() 方法的调用在整个应用上下文中只创建了一个 DataSource 实例,即使它被多次调用。

这对于保证某些 Bean 的单例特性是非常有用的,尤其是当这些 Bean 之间存在依赖关系

2-2、将 proxyBeanMethods 设置为 false

@Configuration(proxyBeanMethods = false)
public class AppConfig {

    // 创建了一个DataSource实例
    @Bean
    public DataSource dataSource() {
        return new HikariDataSource();
    }

    @Bean
    public JdbcTemplate jdbcTemplate() {
        // 这里调用了dataSource()方法来获取DataSource实例
        return new JdbcTemplate(dataSource());
    }
}

在这种情况下,每次从配置类中调用 @Bean 方法(如直接在另一个 @Bean 方法内调用或者在应用中通过上下文调用)都将创建一个新的实例。

这会导致额外的资源消耗和初始化开销,尤其是对设计为单例使用的资源密集型 Bean(如数据库连接池)来说。

但是,这种的好处就是不需要为配置类创建 CGLIB代理,所以可以提升启动时间和减少运行时的内存占用。

探究:不使用 CGLIB 代理为何可以提升启动时间和减少运行时的内存占用?

  • 减少类的加载和处理时间:
    当 proxyBeanMethods 设置为 true 时,Spring 会通过 CGLIB 库为每个配置类创建一个子类(代理类)。

    这个过程需要加载额外的类,执行字节码生成和处理,这会增加应用启动时的CPU和内存开销。

    相反,当 proxyBeanMethods 设置为 false 时,Spring 不会创建这些代理类,从而减少了启动阶段的资源消耗。

  • 避免方法拦截开销:
    在使用 CGLIB 代理的配置类中,每次调用 @Bean 标注的方法时,都会通过代理类,这个代理类会检查是否已经为该 Bean 创建了实例,并确保返回的是同一个实例(单例 Bean)。这个检查和拦截过程在运行时会产生额外的开销

    而 proxyBeanMethods=false 意味着 @Bean 方法会被直接调用,不经过代理类,从而避免了这部分开销。

  • 减少内存占用
    减少代理对象的内存占用:为配置类创建代理对象会增加 JVM 堆内存的使用。

    每个代理类实例都需要消耗一定的内存,特别是在大型应用中,可能有大量的配置类,这种内存占用就变得非常可观。

    关闭代理( proxyBeanMethods=false )意味着这部分内存开销可以被省略。

  • 优化 Spring 容器的管理开销:
    不使用 CGLIB 代理简化了 Spring 容器对 Bean 生命周期的管理。

    Spring 容器不需要维护额外的信息来处理配置类的代理逻辑,这可以进一步降低内存使用并简化容器的运行时行为。

三、何时使用 Full 模式与 Lite 模式?

Full 模式适用于复杂的配置场景,其中配置类中的 Bean 之间存在依赖关系,需要确保通过配置类方法调用获取到的 Bean 是单例的。它适合于那些需要充分利用 Spring 框架提供的依赖注入和 Bean 生命周期管理特性的应用。

Lite 模式适用于简单或性能敏感的应用,其中配置类主要用于组织 Bean 定义,而 Bean 之间的依赖关系较为简单或可以通过其他方式(如构造器注入)解决。

总之,proxyBeanMethods属性的设置取决于你的具体需求,是否需要在配置类中保持@Bean方法调用的单例特性。

在大多数情况下,默认的 true 值是合理的,但在某些高性能需求和简单配置的场景下,将其设置为 false 可能会更有利。

四、总结

Full ( proxyBeanMethods = true ) 保证每个 @Bean 方法被调用多少次返回的组件都是单实例的。

Lite ( proxyBeanMethods = false ) 每个 @Bean 方法被调用多少次返回的组件都是新创建的。

存在组件依赖关系则使用 Full 模式(默认)。否则可以使用 Lite 模式。


finally

如果大家觉得本文写得不错,别忘了给个赞哦!同时,如果您有任何疑问或建议,欢迎在评论区留言,让我们一起交流、探讨!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【开源】JAVA+Vue.js实现在线课程教学系统

目录 一、摘要1.1 系统介绍1.2 项目录屏 二、研究内容2.1 课程类型管理模块2.2 课程管理模块2.3 课时管理模块2.4 课程交互模块2.5 系统基础模块 三、系统设计3.1 用例设计3.2 数据库设计 四、系统展示4.1 管理后台4.2 用户网页 五、样例代码5.1 新增课程类型5.2 网站登录5.3 课…

计算机速成课Crash Course - 30. 万维网

今天继续计算机速成课Crash Course的系列讲解。 更多技术文章,全网首发公众号 “摸鱼IT” 锁定 -上午11点 - ,感谢大家关注、转发、点赞! 计算机速成课Crash Course - 30. 万维网 (qq.com) 30. 万维网 前两集我们深入讨论了电线、信号、交…

【DC渗透系列】DC-4靶场

主机发现 arp-scan -l┌──(root㉿kali)-[~] └─# arp-scan -l Interface: eth0, type: EN10MB, MAC: 00:0c:29:6b:ed:27, IPv4: 192.168.100.251 Starting arp-scan 1.10.0 with 256 hosts (https://github.com/royhills/arp-scan) 192.168.100.1 00:50:56:c0:00:08 …

Gateway API 实践之(八)FSM Gateway SSL 代理终端与 TLS 上游

FSM Gateway 流量管理策略系列: 故障注入黑白名单访问控制限速重试会话保持健康检查负载均衡算法TLS 上游双向 TLS 网关使用 HTTP 对外与客户端通信,而与上游服务使用 HTTPS 的功能,是一种常见的网络架构模式。在这种模式下,网关…

跟着cherno手搓游戏引擎【23】项目维护、2D引擎之前的一些准备

项目维护: 修改文件结构: 头文件自己改改就好了 创建2DRendererLayer: Sandbox2D.h: #pragma once #include "YOTO.h" class Sandbox2D :public YOTO::Layer {public:Sandbox2D();virtual ~Sandbox2D() default;virtual void O…

阿里云服务器租用价格表_2024一年_1个月_1小时收费价格表

2024年阿里云服务器租用价格表更新,云服务器ECS经济型e实例2核2G、3M固定带宽99元一年、ECS u1实例2核4G、5M固定带宽、80G ESSD Entry盘优惠价格199元一年,轻量应用服务器2核2G3M带宽轻量服务器一年61元、2核4G4M带宽轻量服务器一年165元12个月、2核4G服…

LabVIEW工业监控系统

LabVIEW工业监控系统 介绍了一个基于LabVIEW软件开发的工业监控系统。系统通过虚拟测控技术和先进的数据处理能力,实现对工业过程的高效监控,提升系统的自动化和智能化水平,从而满足现代工业对高效率、高稳定性和低成本的需求。 随着工业自…

时间序列预测 —— DeepAR 模型

时间序列预测 —— DeepAR 模型 DeepAR 模型是一种专门用于处理时间序列概率预测的深度学习模型,它可以自动学习数据中的复杂模式,提高预测的准确性。本文将介绍 DeepAR 模型的理论基础、优缺点,并通过 Python 实现单步预测和多步预测的完整…

51单片机编程应用(C语言):串口通信

目录 通信的基本概念和种类 1.1串行通信与并行通信 ​编辑 1.2同步通信与异步通信 1.3单工,半双工,全双工 1.4通信速率 二、波特率和比特率的关系 串口通信简介: 1.接口标准 RS-232 2、D型9针接口定义 3.通信协议: …

金和OA C6 RssModulesHttp.aspx SQL注入漏洞复现

0x01 产品简介 金和网络是专业信息化服务商,为城市监管部门提供了互联网+监管解决方案,为企事业单位提供组织协同OA系统开发平台,电子政务一体化平台,智慧电商平台等服务。 0x02 漏洞概述 金和OA C6 RssModulesHttp.aspx接口处存在SQL注入漏洞,攻击者除了可以利用 SQL 注入…

018 Linux

文章目录 操作系统定义分类Linux系统构成 Linux文件系统Linux常用命令基础操作命令文件操作压缩解压权限管理显示展示命令其他命令 vi编译器操作使用 添加用户基本概念用户管理命令 ubuntu软件安装ssh服务终端启动Python服务 操作系统 定义 操作系统是管理计算机硬件与软件资…

【Linux系统学习】 4.Linux实用操作 上

Linux实用操作 1.各类小技巧(快捷键) 1.1 ctrl c 强制停止 Linux某些程序的运行,如果想要强制停止它,可以使用快捷键ctrl c 命令输入错误,也可以通过快捷键ctrl c,退出当前输入,重新输入 1…

前端JavaScript篇之如何获得对象非原型链上的属性?

目录 如何获得对象非原型链上的属性? 如何获得对象非原型链上的属性? 要获取对象上非原型链上的属性,可以使用 hasOwnProperty() 方法。这个方法是 JavaScript 内置的对象方法,用于检查一个对象是否包含指定名称的属性&#xff0…

数字孪生与智慧园区的融合:打造未来产业生态的新篇章

随着科技的飞速发展,数字孪生和智慧园区已经成为当今社会发展的重要趋势。数字孪生技术为物理世界的对象提供了数字化的复制体,而智慧园区则通过各种信息技术手段实现园区的智能化管理。二者的融合,将为未来产业生态的发展开辟新的篇章。 一…

大华智慧园区综合管理平台 deleteFtp RCE漏洞复现

免责声明:文章来源互联网收集整理,请勿利用文章内的相关技术从事非法测试,由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失,均由使用者本人负责,所产生的一切不良后果与文章作者无关。该…

LLM大语言模型(六):RAG模式下基于PostgreSQL pgvector插件实现vector向量相似性检索

目录 HightLightMac上安装PostgreSQLDBever图形界面管理端创建DB 使用向量检索vector相似度计算近似近邻索引HNSW近似近邻索引示例 HightLight 使用PostgreSQL来存储和检索vector,在数据规模非庞大的情况下,简单高效。 可以和在线业务共用一套DB&#…

【PyQt】08 - 编辑Tab顺序

文章目录 前言一、Tab顺序二、编辑Tab顺序总结 前言 介绍了什么是Tab顺序,以及如何修改Tab顺序。 一、Tab顺序 当你的界面设计好之后,在输入栏按住Tab按键,他会按照你摆放的顺序一次转跳 二、编辑Tab顺序 方法一 然后鼠标左击就可以改变…

【力扣】快乐数,哈希集合 + 快慢指针 + 数学

快乐数原题地址 方法一:哈希集合 定义函数 getNext(n) ,返回 n 的所有位的平方和。一直执行 ngetNext(n) ,最终只有 2 种可能: n 停留在 1 。无限循环且不为 1 。 证明:情况 1 是存在的,如力扣的示例一…

AcWing 1224 交换瓶子(简单图论)

[题目概述] 有 N 个瓶子,编号 1∼N,放在架子上。 比如有 5 个瓶子: 2 1 3 5 4 要求每次拿起 2 个瓶子,交换它们的位置。 经过若干次后,使得瓶子的序号为: 1 2 3 4 5 对于这么简单的情况,显然&a…

Linux命令行工具使用HTTP代理的方法详解

亲爱的Linux用户们,有没有想过在命令行世界里,你的每一个指令都能悄无声息地穿越千山万水,而不被外界窥探?哈哈,没错,就是通过HTTP代理!今天,我们就来一起探索如何在Linux命令行工具…