重看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

相关文章

Java五子棋之坐标校正

上篇针对了Java项目中的解构思维,在这篇内容中我们不妨从整体项目中拆解拿出一个非常重要的五子棋逻辑实现:坐标校正,我们如何使漫无目的鼠标点击变得有序化和可控化呢? 目录 一、从鼠标监听到获取坐标 1.MouseListener和MouseAdapter 2.mousePressed方法 二、坐标校正的具体实现方法 1.关于fillOval方法 2.坐标获取 3.坐标转换 4.坐

Spring Cloud:构建分布式系统的利器

引言 在当今的云计算和微服务架构时代,构建高效、可靠的分布式系统成为软件开发的重要任务。Spring Cloud 提供了一套完整的解决方案,帮助开发者快速构建分布式系统中的一些常见模式(例如配置管理、服务发现、断路器等)。本文将探讨 Spring Cloud 的定义、核心组件、应用场景以及未来的发展趋势。 什么是 Spring Cloud Spring Cloud 是一个基于 Spring

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J

java8的新特性之一(Java Lambda表达式)

1:Java8的新特性 Lambda 表达式: 允许以更简洁的方式表示匿名函数(或称为闭包)。可以将Lambda表达式作为参数传递给方法或赋值给函数式接口类型的变量。 Stream API: 提供了一种处理集合数据的流式处理方式,支持函数式编程风格。 允许以声明性方式处理数据集合(如List、Set等)。提供了一系列操作,如map、filter、reduce等,以支持复杂的查询和转

Java面试八股之怎么通过Java程序判断JVM是32位还是64位

怎么通过Java程序判断JVM是32位还是64位 可以通过Java程序内部检查系统属性来判断当前运行的JVM是32位还是64位。以下是一个简单的方法: public class JvmBitCheck {public static void main(String[] args) {String arch = System.getProperty("os.arch");String dataM

详细分析Springmvc中的@ModelAttribute基本知识(附Demo)

目录 前言1. 注解用法1.1 方法参数1.2 方法1.3 类 2. 注解场景2.1 表单参数2.2 AJAX请求2.3 文件上传 3. 实战4. 总结 前言 将请求参数绑定到模型对象上,或者在请求处理之前添加模型属性 可以在方法参数、方法或者类上使用 一般适用这几种场景: 表单处理:通过 @ModelAttribute 将表单数据绑定到模型对象上预处理逻辑:在请求处理之前

eclipse运行springboot项目,找不到主类

解决办法尝试了很多种,下载sts压缩包行不通。最后解决办法如图: help--->Eclipse Marketplace--->Popular--->找到Spring Tools 3---->Installed。

JAVA读取MongoDB中的二进制图片并显示在页面上

1:Jsp页面: <td><img src="${ctx}/mongoImg/show"></td> 2:xml配置: <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001

Java面试题:通过实例说明内连接、左外连接和右外连接的区别

在 SQL 中,连接(JOIN)用于在多个表之间组合行。最常用的连接类型是内连接(INNER JOIN)、左外连接(LEFT OUTER JOIN)和右外连接(RIGHT OUTER JOIN)。它们的主要区别在于它们如何处理表之间的匹配和不匹配行。下面是每种连接的详细说明和示例。 表示例 假设有两个表:Customers 和 Orders。 Customers CustomerIDCus

22.手绘Spring DI运行时序图

1.依赖注入发生的时间 当Spring loC容器完成了 Bean定义资源的定位、载入和解析注册以后,loC容器中已经管理类Bean 定义的相关数据,但是此时loC容器还没有对所管理的Bean进行依赖注入,依赖注入在以下两种情况 发生: 、用户第一次调用getBean()方法时,loC容器触发依赖注入。 、当用户在配置文件中将<bean>元素配置了 lazy-init二false属性,即让