深入理解BeanDefinition和Spring Beans

news/2024/4/26 23:53:13/文章来源:https://blog.csdn.net/lsoxvxe/article/details/131999317

深入理解BeanDefinition和Spring Beans

引言

在Spring框架中,BeanDefinition和Spring Beans是非常重要的概念。BeanDefinition定义了Spring Bean的元数据,而Spring Beans是应用程序中的对象实例。理解BeanDefinition和Spring Beans的概念和使用方法对于开发和维护Spring应用程序非常重要。

本篇博客将深入探讨BeanDefinition和Spring Beans的概念、创建过程、属性详解以及使用BeanDefinition进行动态注册、延迟加载和依赖注入等方面的内容。

什么是BeanDefinition

BeanDefinition是Spring框架中的一个重要概念,它定义了Spring Bean的元数据信息。通过BeanDefinition,可以指定Bean的类名、作用域、构造函数参数、属性值等信息。

BeanDefinition接口定义了以下常用方法:

  • getBeanClassName():获取Bean的类名
  • getScope():获取Bean的作用域
  • setScope(String scope):设置Bean的作用域
  • getPropertyValues():获取Bean的属性值
  • setPropertyValue(String name, Object value):设置Bean的属性值
  • getConstructorArgumentValues():获取构造函数参数值
  • setConstructorArgumentValue(int index, Object value):设置构造函数参数值

Spring Beans的创建过程

Spring Beans的创建过程包括BeanDefinition的解析和实例化两个阶段。

首先,Spring容器会解析配置文件或注解,将Bean的定义转化为对应的BeanDefinition对象。然后,根据BeanDefinition的信息,通过反射机制实例化Bean对象,并进行属性注入和初始化。

在BeanDefinition解析阶段,Spring容器会读取配置文件或扫描注解,将Bean的定义转化为BeanDefinition对象。BeanDefinition对象包含了Bean的类名、作用域、构造函数参数、属性值等信息。

在Bean实例化阶段,Spring容器根据BeanDefinition的信息,通过反射机制实例化Bean对象。然后,Spring容器会根据BeanDefinition中的属性值进行属性注入,并调用Bean的初始化方法。

BeanDefinition的属性详解

BeanDefinition中常用的属性包括bean的类名、作用域、构造函数参数等。

  • Bean的类名:通过getBeanClassName()方法获取Bean的类名。通过设置Bean的类名,Spring容器可以根据类名进行反射实例化Bean对象。

  • 作用域:通过getScope()setScope(String scope)方法获取和设置Bean的作用域。常用的作用域有单例(singleton)和原型(prototype)两种。单例作用域表示Spring容器中只有一个Bean实例,而原型作用域表示每次请求都会创建一个新的Bean实例。

  • 构造函数参数:通过getConstructorArgumentValues()setConstructorArgumentValue(int index, Object value)方法获取和设置构造函数参数值。构造函数参数值可以是基本类型、引用类型或其他Bean。

  • 属性值:通过getPropertyValues()setPropertyValue(String name, Object value)方法获取和设置Bean的属性值。属性值可以是基本类型、引用类型或其他Bean。

使用BeanDefinition进行动态注册Bean

使用BeanDefinition可以在运行时动态注册Bean到Spring容器中。动态注册Bean可以灵活地根据需要创建和管理Bean对象。

下面是一个示例代码,演示如何使用BeanDefinition进行动态注册Bean:

//```java
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.context.support.GenericApplicationContext;public class DynamicBeanRegistrationExample {public static void main(String[] args) {// 创建一个GenericApplicationContext对象GenericApplicationContext context = new GenericApplicationContext();// 获取DefaultListableBeanFactory对象DefaultListableBeanFactory beanFactory = (DefaultListableBeanFactory) context.getBeanFactory();// 创建一个BeanDefinitionBuilder对象BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(MyBean.class);// 设置Bean的属性值builder.addPropertyReference("anotherBean", "anotherBean");// 注册BeanDefinition到BeanFactorybeanFactory.registerBeanDefinition("myBean", builder.getBeanDefinition());// 启动Spring应用上下文context.refresh();// 从容器中获取动态注册的BeanMyBean myBean = (MyBean) context.getBean("myBean");myBean.doSomething();}static class MyBean {private AnotherBean anotherBean;public void setAnotherBean(AnotherBean anotherBean) {this.anotherBean = anotherBean;}public void doSomething() {System.out.println("Doing something with anotherBean: " + anotherBean);}}static class AnotherBean {// ...}
}

上述示例代码中,我们首先创建了一个GenericApplicationContext对象,然后获取其对应的DefaultListableBeanFactory对象。接着,我们使用BeanDefinitionBuilder创建一个MyBeanBeanDefinition,并设置其属性值。最后,我们将BeanDefinition注册到BeanFactory中,并启动应用上下文。通过context.getBean()方法,我们可以从容器中获取动态注册的Bean,并调用其方法。

使用BeanDefinition进行动态注册Bean可以使我们在运行时根据需要创建和管理Bean对象,提供了更大的灵活性。

使用BeanDefinition进行Bean的延迟加载

BeanDefinition可以用于实现Bean的延迟加载,即在需要使用Bean时才进行实例化和初始化。通过延迟加载,可以提高应用程序的性能和资源利用率。

下面是一个示例代码,演示如何使用BeanDefinition实现Bean的延迟加载:

import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.context.support.GenericApplicationContext;public class LazyLoadingExample {public static void main(String[] args) {// 创建一个GenericApplicationContext对象GenericApplicationContext context = new GenericApplicationContext();// 获取DefaultListableBeanFactory对象DefaultListableBeanFactory beanFactory = (DefaultListableBeanFactory) context.getBeanFactory();// 创建一个BeanDefinitionBuilder对象BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(MyBean.class);// 设置Bean的属性值builder.addPropertyReference("anotherBean", "anotherBean");// 设置Bean的延迟加载属性builder.setLazyInit(true);// 注册BeanDefinition到BeanFactorybeanFactory.registerBeanDefinition("myBean", builder.getBeanDefinition());// 启动Spring应用上下文context.refresh();// 在需要使用Bean时,从容器中获取Bean并调用方法MyBean myBean = (MyBean) context.getBean("myBean");myBean.doSomething();}static class MyBean {private AnotherBean anotherBean;public void setAnotherBean(AnotherBean anotherBean) {this.anotherBean = anotherBean;}public void doSomething() {System.out.println("Doing something with anotherBean: " + anotherBean);}}static class AnotherBean {// ...}
}

上述示例代码中,我们在创建MyBeanBeanDefinition时,通过builder.setLazyInit(true)设置了Bean的延迟加载属性为true。这样,在启动应用上下文时,MyBean并不会立即被实例化和初始化,只有在需要使用它时才会进行实例化和初始化。

通过使用BeanDefinition的延迟加载属性,可以避免在应用启动时加载大量的Bean,从而提高应用程序的启动性能和资源利用率。

使用BeanDefinition进行Bean的依赖注入

BeanDefinition可以用于实现Bean的依赖注入,即在创建Bean时自动注入其他Bean的引用。通过依赖注入,可以实现组件之间的松耦合和高内聚。

下面是一个示例代码,演示如何使用BeanDefinition实现Bean的依赖注入:

import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.context.support.GenericApplicationContext;public class DependencyInjectionExample {public static void main(String[] args) {// 创建一个GenericApplicationContext对象GenericApplicationContext context = new GenericApplicationContext();// 获取DefaultListableBeanFactory对象DefaultListableBeanFactory beanFactory = (DefaultListableBeanFactory) context.getBeanFactory();// 创建一个BeanDefinitionBuilder对象BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(MyBean.class);// 设置Bean的属性值builder.addPropertyReference("anotherBean", "anotherBean");// 注册BeanDefinition到BeanFactorybeanFactory.registerBeanDefinition("myBean", builder.getBeanDefinition());// 注册另一个BeanbeanFactory.registerSingleton("anotherBean", new AnotherBean());// 启动Spring应用上下文context.refresh();// 从容器中获取Bean并调用方法MyBean myBean = (MyBean) context.getBean("myBean");myBean.doSomething();}static class MyBean {private AnotherBean anotherBean;public void setAnotherBean(AnotherBean anotherBean) {this.anotherBean = anotherBean;}public void doSomething() {System.out.println("Doing something with anotherBean: " + anotherBean);}}static class AnotherBean {// ...}
}

上述示例代码中,我们在创建MyBeanBeanDefinition时,通过builder.addPropertyReference("anotherBean", "anotherBean")设置了MyBean的属性anotherBean的引用为anotherBean。然后,我们通过beanFactory.registerSingleton("anotherBean", new AnotherBean())注册了另一个Bean。在启动应用上下文后,MyBean会自动获取anotherBean的引用,并可以使用它。

通过使用BeanDefinition进行依赖注入,可以实现Bean之间的解耦和灵活的组件配置。

BeanDefinition的扩展和自定义

除了使用Spring提供的标准的BeanDefinition之外,我们还可以扩展和自定义BeanDefinition,以满足特定的需求。

下面是一个示例代码,演示如何扩展和自定义BeanDefinition:

import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.context.support.GenericApplicationContext;public class CustomBeanDefinitionExample {public static void main(String[] args) {// 创建一个GenericApplicationContext对象GenericApplicationContext context = new GenericApplicationContext();// 获取DefaultListableBeanFactory对象DefaultListableBeanFactory beanFactory = (DefaultListableBeanFactory) context.getBeanFactory();// 创建一个自定义的BeanDefinitionCustomBeanDefinitionBuilder builder = CustomBeanDefinitionBuilder.genericBeanDefinition(MyBean.class);// 设置自定义的属性builder.setCustomProperty("customPropertyValue");
```java// 注册BeanDefinition到BeanFactorybeanFactory.registerBeanDefinition("myBean", builder.getBeanDefinition());// 启动Spring应用上下文context.refresh();// 从容器中获取Bean并调用方法MyBean myBean = (MyBean) context.getBean("myBean");myBean.doSomething();}static class MyBean {// ...}static class CustomBeanDefinitionBuilder extends BeanDefinitionBuilder {private String customProperty;private CustomBeanDefinitionBuilder(Class<?> beanClass) {super(beanClass);}public static CustomBeanDefinitionBuilder genericBeanDefinition(Class<?> beanClass) {return new CustomBeanDefinitionBuilder(beanClass);}public CustomBeanDefinitionBuilder setCustomProperty(String customProperty) {this.customProperty = customProperty;return this;}@Overridepublic BeanDefinition getBeanDefinition() {BeanDefinition beanDefinition = super.getBeanDefinition();// 添加自定义的属性到BeanDefinitionbeanDefinition.setAttribute("customProperty", customProperty);return beanDefinition;}}
}

上述示例代码中,我们创建了一个自定义的CustomBeanDefinitionBuilder,继承自BeanDefinitionBuilder。在CustomBeanDefinitionBuilder中,我们添加了一个自定义的属性customProperty,并重写了getBeanDefinition()方法,将自定义属性添加到BeanDefinition的attributes中。

通过扩展和自定义BeanDefinition,我们可以根据具体需求添加自定义属性或行为,以满足特定的业务场景。

总结

本篇博客深入理解了BeanDefinition和Spring Beans的概念、创建过程、属性详解以及使用BeanDefinition进行动态注册、延迟加载和依赖注入等方面的内容。

通过学习和掌握BeanDefinition,我们可以更好地理解和使用Spring框架,实现灵活、高效的应用程序开发。

希望本篇博客对您理解BeanDefinition和Spring Beans有所帮助,谢谢阅读!

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

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

相关文章

mysql(六)读写分离

目录 前言 一、概述 二、配置JAVA环境及安装阿米巴 三、配置阿米巴 四、测试 总结 前言 MySQL读写分离是一种常见的数据库架构设计模式&#xff0c;旨在提高系统的性能和可用性。它通过将读操作和写操作分离到不同的MySQL实例上来实现。 读写分离的原理是&#xff1a; 1. 主…

UE5.1.1 创建C++项目失败

因一直使用Unity开发环境&#xff0c;安装Unreal后&#xff0c;并未详细配置过其开发环境&#xff0c;默认创建蓝图工程无异常&#xff0c;但创建UE C项目时总共遇到两个错误&#xff1a; 错误一 Running /Epic/UE/UE_5.1/Engine/Build/BatchFiles/Build.bat -projectfiles -…

郑州主域名和多个子域名的泛域名https证书

随着网络的发展&#xff0c;一个站长往往会创建多个域名网站&#xff0c;为了与主站有联系&#xff0c;站长会选择主站域名的二级域名来创建网站&#xff0c;比如主站是www .***. com&#xff0c;那么新创建的网站的域名就是top .***. com、vip .***. com等。 1.在安装部署htt…

原生求生记:揭秘UniApp的原生能力限制

文章目录 1. 样式适配问题2. 性能问题3. 原生能力限制4. 插件兼容性问题5. 第三方组件库兼容性问题6. 全局变量污染7. 调试和定位问题8. 版本兼容性问题9. 前端生态限制10. 文档和支持附录&#xff1a;「简历必备」前后端实战项目&#xff08;推荐&#xff1a;⭐️⭐️⭐️⭐️…

利用小波包对一维信号进行降噪或压缩(MATLAB)

function [ output_args ] example4_12( input_args ) %EXAMPLE4_12 Summary of this function goes here % Detailed explanation goes here clc; clear; % 设置信噪比和随机数的初始值 snr 3; init 2055615866; % 生成一个原始信号xref和含高斯白噪声的信号x [xref,x] …

网络安全/信息安全(黑客技术)自学笔记

一、网络安全基础知识 1.计算机基础知识 了解了计算机的硬件、软件、操作系统和网络结构等基础知识&#xff0c;可以帮助您更好地理解网络安全的概念和技术。 2.网络基础知识 了解了网络的结构、协议、服务和安全问题&#xff0c;可以帮助您更好地解决网络安全的原理和技术…

2023年的深度学习入门指南(22) - 百川大模型13B的运行及量化

2023年的深度学习入门指南(22) - 百川大模型13B的运行及量化 不知道上一讲的大段代码大家看晕了没有。但是如果你仔细看了会发现&#xff0c;其实代码还是不全的。比如分词器我们就没讲。 另外&#xff0c;13B比7B的改进点也没有讲。 再有&#xff0c;对于13B需要多少显存我们…

记录每日LeetCode 141.环形链表 Java实现

题目描述&#xff1a; 给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内部使用整数 pos 来表示链表尾连接到链…

【RTT驱动框架分析00】-应用层函数调用流程-串口为例

应用层函数调用流程分析以-串口为例 1. rt_device_find调用流程 RTT内部有一个全局变量数组rt_object_container数组的数量就是 rt_object_info_type 内核对象的种类&#xff0c;其中包含一个RT_USING_DEVICE (设备的数据类型)&#xff0c;数组内部的每一个变量包含一个链表&…

基于扩展(EKF)和无迹卡尔曼滤波(UKF)的电力系统动态状态估计(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

C\C++内存管理

目录 1.C/C内存分布2.C语言中动态内存管理方式3.C中动态内存管理3.1new/delete内置类型3.2new和delete操作自定义类型 4.operator new与operator delete函数4.2重载operator new与operator delete&#xff08;了解&#xff09; 5.new和delete的实现原理5.1内置类型5.2 自定义类…

【动态规划part12】| 309.买卖股票的最佳时机含冷冻期、714.买卖股票的最佳时机含手续费

目录 &#x1f388;LeetCode309.最佳买卖股票时机含冷冻期 &#x1f388;LeetCode714.买卖股票的最佳时机含手续费 &#x1f388;LeetCode309.最佳买卖股票时机含冷冻期 给定一个整数数组prices&#xff0c;其中第 prices[i] 表示第 i 天的股票价格 。​ 设计一个算法计…

【计算机视觉中的 GAN 】如何稳定GAN训练(3)

一、说明 在上一篇文章中&#xff0c;我们达到了理解未配对图像到图像翻译的地步。尽管如此&#xff0c;在实现自己的超酷深度GAN模型之前&#xff0c;您必须了解一些非常重要的概念。如本文所提的GAN模型新成员的引入&#xff1a;Wasserstein distance&#xff0c;boundary eq…

element时间选择器的默认值

概览&#xff1a;vue使用element组件&#xff0c;需要给时间选择器设置默认值&#xff0c;场景一&#xff1a;默认时间选择器&#xff0c;场景二&#xff1a;时间范围选择器&#xff0c;开始时间和结束时间。 一、默认时间选择器 实现思路&#xff1a; element组件的v-model绑…

DevOps-Jenkins

Jenkins Jenkins是一个可扩展的持续集成引擎&#xff0c;是一个开源软件项目&#xff0c;旨在提供一个开放易用的软件平台&#xff0c;使软件的持续集成变成可能。 官网 应用场景 场景一 研发人员上传开发好的代码到github代码仓库需要将代码下载nginx服务器部署手动下载再…

数值线性代数:奇异值分解SVD

本文记录计算矩阵奇异值分解SVD的原理与流程。 注1&#xff1a;限于研究水平&#xff0c;分析难免不当&#xff0c;欢迎批评指正。 零、预修 0.1 矩阵的奇异值 设列满秩矩阵&#xff0c;若的特征值为&#xff0c;则称为矩阵的奇异值。 0.2 SVD(分解)定理 设&#xff0c;则…

Open3D-ML自动驾驶点云目标检测与分割入门

当开始新的研究时&#xff0c;我的方法通常是测试不同的相关事物&#xff0c;直到有足够的经验让我开始将这些点联系起来。 在开始构建用于 3D 对象检测的自定义模型之前&#xff0c;我购买了一台 LiDAR 并处理了一些数据。 下一个明显的步骤是在我为自己的数据贴标签之前找出研…

SpringCloudAlibaba之Ribbon

Ribbon是nacos自带的负载均衡器&#xff0c;属于客户端的负载均衡 但是在Spring高级版本中让LoadBalancer替代了 本人用的是2.1.0的nacos&#xff0c;ribbon还没有被替换。 使用&#xff1a; 在配置类中&#xff1a;LoadBalanced BeanLoadBalancedpublic RestTemplate restT…

ChatGPT长文本对话输入方法

ChatGPT PROMPTs Splitter 是一个开源工具&#xff0c;旨在帮助你将大量上下文数据分成更小的块发送到 ChatGPT 的提示&#xff0c;并根据如何处理所有块接收到 ChatGPT&#xff08;或其他具有字符限制的语言模型&#xff09;的方法。 推荐&#xff1a;用 NSDT设计器 快速搭建可…

【FPGA IP系列】FIFO的通俗理解

FPGA厂商提供了丰富的IP核&#xff0c;基础性IP核都是可以直接免费调用的&#xff0c;比如FIFO、RAM等等。 本文主要介绍FIFO的一些基础知识&#xff0c;帮助大家能够理解FIFO的基础概念。 一、FIFO介绍 FIFO全称是First In First Out&#xff0c;即先进先出。 FIFO是一个数…