深入理解BeanDefinition和Spring Beans

深入理解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.mzph.cn/news/14180.shtml

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

相关文章

2023牛客暑期多校-J-Qu‘est-ce Que C‘est?(DP)

题意&#xff1a; 给定长度为n的数列,要求每个数都在的范围&#xff0c;且任意长度大于等于2的区间和都大于等于0&#xff0c;问方案数。。 思路&#xff1a; 首先要看出是dp题&#xff0c;用来表示遍历到第i位且后缀和最小为x的可行方案数&#xff08;此时的后缀可以只有最…

IT技术面试必备:如何做好IT类技术面试?

博主 默语带您 Go to New World. ✍ 个人主页—— 默语 的博客&#x1f466;&#x1f3fb; 《java 面试题大全》 &#x1f369;惟余辈才疏学浅&#xff0c;临摹之作或有不妥之处&#xff0c;还请读者海涵指正。☕&#x1f36d; 《MYSQL从入门到精通》数据库是开发者必会基础之…

oracle建立自动增长字段

oracle数据库与其他的数据库不太一样&#xff0c;比如在mysql里自动增长只要设定“auto_increment”即可。可是在oracle里就没有这种配置了。以oracle11g为例&#xff0c;建立自动增长的字段。操作如下&#xff1a; --创建表 create table USERINFO ( ID NUMBER , …

【Linux】多线程的补充

1 线程安全的单例模式 1.1 什么是单例模式 单例模式是一种 "经典的, 常用的, 常考的" 设计模式. 1.2 什么是设计模式 IT行业这么火, 涌入的人很多. 俗话说林子大了啥鸟都有. 大佬和菜鸡们两极分化的越来越严重. 为了让菜鸡们不太拖大佬的后腿, 于是大佬们针对一些…

开发经验分享之:import引入包和@Autowired注入类有什么区别

大家好&#xff0c;我是三叔&#xff0c;很高兴这期又和大家见面了&#xff0c;一个奋斗在互联网的打工人。 import 和 Autowired 想必大家在 Java 开发中使用频率最多的关键字之一了把&#xff0c;这篇博客将解释这两个概念的区别和作用&#xff0c;帮助你更好地理解它们在Ja…

如何开发一套苹果cms前端模板

本文运用了苹果cms官网的模板开发教程&#xff0c;开发了一套苹果cms的前端模板&#xff0c;感兴趣的同学可以去github下载使用。 什么是模板 模板是网站的主题外观&#xff0c;也被称为主题或皮肤。通过使用不同的模板&#xff0c;网站的前台可以以不同的样式展示。这就像人们…

04-导数判断凹(concave)凸(convex)性_导数用于泰勒展开

导数与函数凹凸性的关系 函数的二阶导数是和函数的凹凸性是有关系的&#xff0c;凹凸性怎么定义的&#xff1f; 先来做简单的回顾&#xff0c;更多的会在最优化方法里面给大家讲&#xff0c;这里先记住凸函数是向下凸的&#xff0c; 反正就是凹的&#xff0c;是否是凸函数可以…

秒级体验本地调试远程 k8s 中的服务

点击上方蓝色字体&#xff0c;选择“设为星标” 回复”云原生“获取基础架构实践 背景 在这个以k8s为云os的时代&#xff0c;程序员在日常的开发过程中&#xff0c;肯定会遇到各种问题&#xff0c;比如&#xff1a;本地开发完&#xff0c;需要部署到远程k8s集群&#xff0c;本地…

【设计模式】详解观察者模式

文章目录 1、简介2、观察者模式简单实现抽象主题&#xff08;Subject&#xff09;具体主题&#xff08;ConcreteSubject&#xff09;抽象观察者&#xff08;Observer&#xff09;具体观察者&#xff08;ConcrereObserver&#xff09;测试&#xff1a; 观察者设计模式优缺点观察…

01Mysql查询语句题型总结

#第三章 SELECT课后练习 # 1.查询员工12个月的工资总和&#xff0c;并起别名为ANNUAL SALARY SELECT employee_id,last_name,salary * 12 "ANNUAL SALARY" FROM employees; #加奖金的情况 SELECT employee_id,last_name,salary * 12 *(1 IFNULL(commission_pct,0)) …

音视频(关于视频的封装,由mp4->mov)

(20条消息) FFmpeg Visual studio 开发环境搭建_HW140701的博客-CSDN博客 1.封装格式&#xff1a;AVI&#xff0c;MP4&#xff0c;ASF AVI&#xff1a;压缩标准可以任意选 FLV,ts:直播等使用的流媒体 mp4:既是封装又是压缩 #pragma comment(lib,"avcodec.lib") …

十八章:用于弱监督语义分割的自监督等变注意力机制

0.摘要 图像级弱监督语义分割是一个具有挑战性的问题&#xff0c;近年来得到了深入研究。大多数先进的解决方案利用类激活图&#xff08;CAM&#xff09;。然而&#xff0c;由于全监督和弱监督之间存在差距&#xff0c;CAM几乎无法用作对象掩码。在本文中&#xff0c;我们提出了…

深度优先搜索|79, 695,212

深度优先搜索|79. 单词搜索, 695. 岛屿的最大面积, 212. 单词搜索 II 单词搜索岛屿的最大面积单词搜索II 单词搜索 用的是深度优先搜索&#xff0c;这种判断类型的回溯我就一直不知道要怎么回退&#xff0c;然后勉强写了一个。 这里还有一个注意事项就是&#xff0c;走到最后一…

[JAVAee]定时器

目录 定时器的含义 定时器的使用 定时器的解析 ①TaskQueue ​②TimerThread ③Timer 定时器的模拟实现 ①创建Task自定义类型 ②创建TimerThread类 ③Timer类 完整代码 定时器的含义 从名字上看,就是我们通俗理解的那个定时器.设置一定的时间,并在一定的时间后发生…

安卓抓包神奇黄鸟HttpCanary安装配置及使用教程

1、下载安装包 黄鸟抓包下载地址 2、安装下载的apk 3、证书安装问题 vivo手机我安装时打开黄鸟app&#xff0c;会直接弹出&#xff0c;直接安装即可 其他手机&#xff0c;需要去系统设置中安装 3.1 搜索 证书&#xff0c;选择CA证书 3.2 进行本人操作验证 3.3 安装HttpCa…

黄东旭:The Future of Database,掀开 TiDB Serverless 的引擎盖

在 PingCAP 用户峰会 2023 上&#xff0c; PingCAP 联合创始人兼 CTO 黄东旭 分享了“The Future of Database”为主题的演讲&#xff0c; 介绍了 TiDB Serverless 作为未来一代数据库的核心设计理念。黄东旭 通过分享个人经历和示例&#xff0c;强调了数据库的服务化而非服务化…

WPF实战学习笔记24-首页编辑与完成

首页编辑与完成 indexview添加Listbox控件的鼠标双击行为 添加todo、memo的编辑命令indexviewmodel添加对应的更新事件处理 添加ToggleButton与后台的绑定 将ToggleButton的ischeck绑定到status属性添加bool int 转换器添加完成命令添加完成功能函数 Listbox添加行为 给行为…

C语言假期作业 DAY 01

题目 1.选择题 1、执行下面程序&#xff0c;正确的输出是&#xff08; &#xff09; int x5,y7; void swap() { int z; zx; xy; yz; } int main() { int x3,y8; swap(); printf("%d,%d\n"&#xff0c;x, y)…

Android Studio 代码模板插件实现

Android Studio 代码模板插件 背景 可以跳过背景和简述&#xff0c;从模板插件实现开始看. 开发新页面时&#xff0c;原先需要写一堆模板代码。比如用Databinding写列表结构的页面&#xff0c;需要手写以下文件&#xff1a; XxActivity.ktXxFragment.ktXxViewModel.ktXxListA…

一、SpringBoot基础[日志]

一、日志 解释&#xff1a;SpringBoot使用logback作为默认的日志框架&#xff0c;其中还可以导入log4j2等优秀的日志框架 1.修改日志内容 修改整个日志格式&#xff1a;logging.pattern.console%d{yyyy-MM-dd HH:mm:ss} %-5level [%thread] %logger{15} 你好 %msg%n %d{yyy…