重看Spring聚焦BeanDefinition分析和构造

2024-05-02 13:20

本文主要是介绍重看Spring聚焦BeanDefinition分析和构造,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

一、对BeanDefinition的理解

(一)理解元信息

(二)BeanDefinition理解分析

二、BeanDefinition的结构设计分析

(一)整体结构体会

(二)重要接口和类分析

三、构造 BeanDefinition:模式注解 + 组件扫描方式

四、构造 BeanDefinition:配置类 + @Bean 注解方式

五、构造 BeanDefinition:通过XML配置文件

六、构造 BeanDefinition:通过编程方式

七、对构造分析对比


一、对BeanDefinition的理解

BeanDefinition 可以被认为是 Bean 的元信息。它不是 Bean 实例本身,而是描述了如何创建 Bean 实例以及 Bean 的配置信息。因此,BeanDefinition 提供了关于 Bean 的元数据,包括类名、作用域、构造函数参数、属性值等等。在 Spring 容器启动时,BeanDefinition 被解析并用于实例化和管理 Bean 实例。

(一)理解元信息

在计算机科学中,元信息是用于描述其他数据的数据。元信息提供关于数据的属性、结构、类型或其他相关信息,通常用于解释和管理数据,而不是直接操作数据本身。

在软件开发中,元信息的概念非常常见。比如,在数据库中,表的结构信息(如列名、数据类型、约束等)就是元信息;在编程语言中,注解(Annotation)提供了关于类、方法或字段的额外信息,也可以看作是元信息;在 Web 开发中,HTML 标签中的属性、HTTP 头部中的信息等都可以视为元信息。

(二)BeanDefinition理解分析

BeanDefinition 是关于 Bean 的元数据集合(包含了创建和管理 Bean 所需的所有信息),这些信息在 Spring 容器启动时被解析,并用于实例化和管理 Bean 实例。通过配置和操作 BeanDefinition可以控制和定制 Bean 的行为,实现更加灵活和高效的应用程序开发。具体的元信息可总结如下表进行查看:

元信息描述
类信息 (Class Information)Bean的类名,用于指定如何实例化Bean。
作用域信息 (Scope Information)Bean的作用域,决定了Bean的生命周期范围,包括singleton、prototype、request、session等。
构造函数参数 (Constructor Arguments)描述了Bean构造函数的参数信息,用于实例化Bean对象。
属性信息 (Property Information)描述了Bean的属性值信息,包括属性名和属性值,用于在实例化后设置Bean的属性。
初始化和销毁方法 (Initialization and Destruction Methods)指定了Bean的初始化方法和销毁方法,用于在Bean实例化后或销毁前执行额外的操作。
依赖信息 (Dependency Information)描述了Bean之间的依赖关系,使得容器能够按正确的顺序实例化和管理Bean。
其他元信息 (Other Metadata)包括Bean的描述、别名等其他元信息,用于进一步描述Bean的用途和特性。

二、BeanDefinition的结构设计分析

(一)整体结构体会

(二)重要接口和类分析

我们选取一些重要接口和类整理表格来快速回顾(源码暂时不进行分析了):

接口/类描述
BeanDefinition 接口定义了 Bean 的元信息,包括类名、作用域、构造函数参数、属性值等。
AbstractBeanDefinition 抽象类BeanDefinition 接口的抽象实现类,提供了通用属性的默认实现,如 Bean 类型、作用域、懒加载等。
GenericBeanDefinition 类AbstractBeanDefinition 的具体实现类,用于描述通用的 BeanDefinition 结构,适用于多种配置方式。
RootBeanDefinition 类GenericBeanDefinition 的子类,增加了对 Bean 类型自动检测的支持,可以根据配置自动确定 Bean 的类型。
AnnotatedBeanDefinition 接口表示使用注解配置的 BeanDefinition,继承自 BeanDefinition 接口,用于描述通过注解方式配置的 Bean。
BeanDefinitionHolder 类BeanDefinition 的持有者,包含一个 BeanDefinition 对象以及与之关联的名称和别名。
BeanDefinitionRegistry 接口定义了 BeanDefinition 注册的方法,允许向 Spring 容器注册新的 BeanDefinition,或者从容器中移除已有的 BeanDefinition。
DefaultListableBeanFactory 类BeanDefinitionRegistry 接口的默认实现类,实现了 BeanDefinition 的注册和管理功能,是 Spring 容器的核心部分之一。

接下来我们重点看如何构造BeanDefinition。

三、构造 BeanDefinition:模式注解 + 组件扫描方式

基于约定优于配置的原则

使用注解(如 @Component@Service@Repository@Controller 等)标记类,然后通过组件扫描(Component Scanning)方式,Spring 容器会自动扫描指定的包,并根据这些注解自动创建相应的 BeanDefinition。

这种方式在平时开发中最为常见,比如直接定义一个简单的组件类并用 @Component 注解进行标记如下:

package org.zyf.javabasic.spring.beandefinition;import org.springframework.stereotype.Component;/*** @program: zyfboot-javabasic* @description: 定义一个简单的组件类,并使用 @Component 注解进行标记* @author: zhangyanfeng* @create: 2024-05-02 11:03**/
@Component
public class ZyfComponent {public void sayHello() {System.out.println("Hello from ZyfComponent!");}
}

为符合其构造,我们在定义一个配置类并在其中启用组件扫描如下:

package org.zyf.javabasic.spring.beandefinition;import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;/*** @program: zyfboot-javabasic* @description: 定义一个配置类,并在其中启用组件扫描。* @author: zhangyanfeng* @create: 2024-05-02 11:08**/
@Configuration
@ComponentScan("org.zyf.javabasic.spring.beandefinition")
public class ZyfAppConfig {// 配置类的其他内容
}

直接验证如下:Spring 容器在启动时扫描指定包路径下的组件类,并将标记了 @Component 注解的类注册为 BeanDefinition,这样就可以方便地在应用程序中使用这些 Bean。

package org.zyf.javabasic.spring.beandefinition;import org.springframework.context.annotation.AnnotationConfigApplicationContext;/*** @program: zyfboot-javabasic* @description: ceshi* @author: zhangyanfeng* @create: 2024-05-02 11:09**/
public class ZyfApplication {public static void main(String[] args) {AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ZyfAppConfig.class);ZyfComponent myComponent = context.getBean(ZyfComponent.class);myComponent.sayHello();context.close();}
}

四、构造 BeanDefinition:配置类 + @Bean 注解方式

显式配置每个 Bean 的创建过程

创建一个配置类,通常使用 @Configuration 注解标记,在这个类中,使用 @Bean 注解标记方法,方法返回的对象就是一个 Bean。

简单来说,可以先创建一个普通的 Java 类作为需要被 Spring 管理的 Bean:

package org.zyf.javabasic.spring.beandefinition;/*** @program: zyfboot-javabasic* @description: 创建一个普通的 Java 类,作为需要被 Spring 管理的 Bean。* @author: zhangyanfeng* @create: 2024-05-02 11:18**/
public class ZyfService {public void performAction() {System.out.println("Performing action in MyService!");}
}

在刚刚创建的配置类ZyfAppConfig中使用 @Bean 注解标记方法增加即可:

package org.zyf.javabasic.spring.beandefinition;import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;/*** @program: zyfboot-javabasic* @description: 定义一个配置类,并在其中启用组件扫描。* @author: zhangyanfeng* @create: 2024-05-02 11:08**/
@Configuration
@ComponentScan("org.zyf.javabasic.spring.beandefinition")
public class ZyfAppConfig {// 配置类的其他内容@Beanpublic ZyfService zyfService() {return new ZyfService();}
}

直接验证如下:

package org.zyf.javabasic.spring.beandefinition;import org.springframework.context.annotation.AnnotationConfigApplicationContext;/*** @program: zyfboot-javabasic* @description: ceshi* @author: zhangyanfeng* @create: 2024-05-02 11:09**/
public class ZyfApplication {public static void main(String[] args) {AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ZyfAppConfig.class);ZyfComponent myComponent = context.getBean(ZyfComponent.class);myComponent.sayHello();ZyfService zyfService = context.getBean(ZyfService.class);zyfService.performAction();context.close();}
}

返回结果符合预期。

五、构造 BeanDefinition:通过XML配置文件

通过 XML 配置文件构造 BeanDefinition 是 Spring 框架最传统的方式之一。

传统的方式是通过 XML 配置文件来定义 Bean,XML 配置文件中的 <bean> 元素就是描述 BeanDefinition 的方式之一。

现在,我们将之前的ZyfService配置到XML 配置文件中如下:

<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"><!--    <bean id="myTestMsgBean" class="org.zyf.javabasic.springextend.bfppext.MyTestMsgBean">-->
<!--        <property name="message" value="Original Message"/>-->
<!--    </bean>--><!--    <bean class="org.zyf.javabasic.springextend.bfppext.MyBeanFactoryPostProcessorForUpdateBean"/>-->
<!--    <bean class="org.zyf.javabasic.springextend.bfppext.MyBeanFactoryPostProcessorForNewDefinRegi"/>-->
<!--    <bean class="org.zyf.javabasic.springextend.bfppext.MyBeanFactoryPostProcessorForDefineMata"/>--><bean id="zyfService" class="org.zyf.javabasic.spring.beandefinition.ZyfService"/></beans>

使用 ClassPathXmlApplicationContext 类来加载配置文件,从而启动 Spring 容器如下:

package org.zyf.javabasic.spring.beandefinition;import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;/*** @program: zyfboot-javabasic* @description: ceshi* @author: zhangyanfeng* @create: 2024-05-02 11:09**/
public class ZyfApplication {public static void main(String[] args) {AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ZyfAppConfig.class);ZyfComponent myComponent = context.getBean(ZyfComponent.class);myComponent.sayHello();ZyfService zyfService = context.getBean(ZyfService.class);zyfService.performAction();ClassPathXmlApplicationContext contextXml = new ClassPathXmlApplicationContext("zyf_application_context.xml");ZyfService myServiceXml = context.getBean("zyfService", ZyfService.class);myServiceXml.performAction();context.close();}
}

六、构造 BeanDefinition:通过编程方式

通过编程方式构造 BeanDefinition允许开发人员完全控制 BeanDefinition 的创建过程。

我们可以使用 Spring 提供的类(如 GenericBeanDefinitionRootBeanDefinition 等)来创建 BeanDefinition 对象,并指定 Bean 的各种属性和配置信息。

也就是可以直接操作代码即可,在原始代码上直接修改如下:

package org.zyf.javabasic.spring.beandefinition;import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.beans.factory.support.GenericBeanDefinition;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;/*** @program: zyfboot-javabasic* @description: ceshi* @author: zhangyanfeng* @create: 2024-05-02 11:09**/
public class ZyfApplication {public static void main(String[] args) {System.out.println("===================通过模式注解 + 组件扫描方式===================");AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ZyfAppConfig.class);ZyfComponent myComponent = context.getBean(ZyfComponent.class);myComponent.sayHello();System.out.println("===================通过配置类 + @Bean 注解方式===================");ZyfService zyfService = context.getBean(ZyfService.class);zyfService.performAction();System.out.println("===================通过XML配置文件方式===================");ClassPathXmlApplicationContext contextXml = new ClassPathXmlApplicationContext("zyf_application_context.xml");ZyfService myServiceXml = context.getBean("zyfService", ZyfService.class);myServiceXml.performAction();System.out.println("===================通过编程方式===================");// 创建一个 BeanDefinition,并指定类名GenericBeanDefinition beanDefinition = new GenericBeanDefinition();beanDefinition.setBeanClass(ZyfService.class);// 将 BeanDefinition 注册到 Spring 容器中DefaultListableBeanFactory beanFactory = (DefaultListableBeanFactory) context.getBeanFactory();beanFactory.registerBeanDefinition("zyfService", beanDefinition);// 获取 Bean,并调用方法ZyfService myService = context.getBean("zyfService", ZyfService.class);myService.performAction();context.close();}
}

验证打印如下,符合预期。

七、对构造分析对比

方式配置灵活性可读性和维护性适用场景
模式注解 + 组件扫描方式

配置简单,基于约定优于配置的原则

适用于大部分情况

配置简单,但可能导致扫描过多的类,降低可读性

简单的项目或小规模团队

快速开发原型或中小型项目

配置类 + @Bean 注解方式灵活性更高,可以通过代码进行复杂的配置和逻辑处理相对易读,也更易于维护,因为 Bean 的创建过程明确可见

需要更灵活配置的项目

对可维护性要求较高的大型项目

通过 XML 配置文件传统方式,配置直观,但可读性较差相对直观,但随项目规模增大,配置文件可能变得臃肿

传统项目

需要与其他框架整合的场景

通过编程方式灵活性最高,可以完全控制 BeanDefinition 的创建过程-代码量较多,可读性较差,维护成本相对较高

需要动态配置 BeanDefinition 的场景

需要在运行时动态注册 Bean 的场景

参考链接和文章

Java-based Container Configuration :: Spring Framework

Bean Scopes :: Spring Framework

Container Overview :: Spring Framework

这篇关于重看Spring聚焦BeanDefinition分析和构造的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/954338

相关文章

Spring常见错误之Web嵌套对象校验失效解决办法

《Spring常见错误之Web嵌套对象校验失效解决办法》:本文主要介绍Spring常见错误之Web嵌套对象校验失效解决的相关资料,通过在Phone对象上添加@Valid注解,问题得以解决,需要的朋... 目录问题复现案例解析问题修正总结  问题复现当开发一个学籍管理系统时,我们会提供了一个 API 接口去

Java操作ElasticSearch的实例详解

《Java操作ElasticSearch的实例详解》Elasticsearch是一个分布式的搜索和分析引擎,广泛用于全文搜索、日志分析等场景,本文将介绍如何在Java应用中使用Elastics... 目录简介环境准备1. 安装 Elasticsearch2. 添加依赖连接 Elasticsearch1. 创

Spring核心思想之浅谈IoC容器与依赖倒置(DI)

《Spring核心思想之浅谈IoC容器与依赖倒置(DI)》文章介绍了Spring的IoC和DI机制,以及MyBatis的动态代理,通过注解和反射,Spring能够自动管理对象的创建和依赖注入,而MyB... 目录一、控制反转 IoC二、依赖倒置 DI1. 详细概念2. Spring 中 DI 的实现原理三、

SpringBoot 整合 Grizzly的过程

《SpringBoot整合Grizzly的过程》Grizzly是一个高性能的、异步的、非阻塞的HTTP服务器框架,它可以与SpringBoot一起提供比传统的Tomcat或Jet... 目录为什么选择 Grizzly?Spring Boot + Grizzly 整合的优势添加依赖自定义 Grizzly 作为

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Java后端接口中提取请求头中的Cookie和Token的方法

《Java后端接口中提取请求头中的Cookie和Token的方法》在现代Web开发中,HTTP请求头(Header)是客户端与服务器之间传递信息的重要方式之一,本文将详细介绍如何在Java后端(以Sp... 目录引言1. 背景1.1 什么是 HTTP 请求头?1.2 为什么需要提取请求头?2. 使用 Spr

Java如何通过反射机制获取数据类对象的属性及方法

《Java如何通过反射机制获取数据类对象的属性及方法》文章介绍了如何使用Java反射机制获取类对象的所有属性及其对应的get、set方法,以及如何通过反射机制实现类对象的实例化,感兴趣的朋友跟随小编一... 目录一、通过反射机制获取类对象的所有属性以及相应的get、set方法1.遍历类对象的所有属性2.获取

Java中的Opencv简介与开发环境部署方法

《Java中的Opencv简介与开发环境部署方法》OpenCV是一个开源的计算机视觉和图像处理库,提供了丰富的图像处理算法和工具,它支持多种图像处理和计算机视觉算法,可以用于物体识别与跟踪、图像分割与... 目录1.Opencv简介Opencv的应用2.Java使用OpenCV进行图像操作opencv安装j

java Stream操作转换方法

《javaStream操作转换方法》文章总结了Java8中流(Stream)API的多种常用方法,包括创建流、过滤、遍历、分组、排序、去重、查找、匹配、转换、归约、打印日志、最大最小值、统计、连接、... 目录流创建1、list 转 map2、filter()过滤3、foreach遍历4、groupingB

SpringBoot如何使用TraceId日志链路追踪

《SpringBoot如何使用TraceId日志链路追踪》文章介绍了如何使用TraceId进行日志链路追踪,通过在日志中添加TraceId关键字,可以将同一次业务调用链上的日志串起来,本文通过实例代码... 目录项目场景:实现步骤1、pom.XML 依赖2、整合logback,打印日志,logback-sp