【Spring】Bean
Spring
将管理对象称为 Bean
。 Spring
可以看作是一个大型工厂,用于生产和管理 Spring
容器中的 Bean
。如果要使用 Spring
生产和管理 Bean
,那么就需要将 Bean
配置在 Spring
的配置文件中。Spring
框架支持 XML
和 Properties
两种格式的配置文件,在实际开发中常用 XML
格式的配置文件。
Bean 的配置
通常,在 XML
文件中,根元素为 <beans>
, <beans>
中包含多个 <bean>
子元素,每个 <bean>
定义一个 Bean
,并描述 Bean
如何被装配到 Spring
容器中。
<beans><bean/><bean></bean>
</beans>
属性或子元素名称 | 描述 |
---|---|
id | Bean 在 BeanFactory 中的唯一标识,获取 Bean 实例时需要以此作为索引名称 |
class | Bean 的具体实现类,使用类名 |
scope | 指定 Bean 实例的作用域 |
name | Bean 的别名,多个别名之间用逗号、空格或分号分隔 |
<constructor-arg> | <bean> 元素的子元素,使用构造方法注入,指定构造方法的参数。该元素的 index 属性指定参数的序号,ref 属性指定对 BeanFactory 中其他 Bean 的引用关系,type 属性指定参数类型,value 属性指定参数的常量值 |
<property> | <bean> 元素的子元素,用于设置一个属性。该元素的 name 属性指定 Bean 实例中相应的属性名称,value 属性指定 Bean 的属性值,ref 属性指定属性对 BeanFactory 中其他 Bean 的引用关系 |
<list> | <property> 元素的子元素,用于封装 List 或数组类型的依赖注入 |
<map> | <property> 元素的子元素,用于封装 Map 类型的依赖注入 |
<set> | <property> 元素的子元素,用于封装 Set 类型的依赖注入 |
<entry> | <map> 元素的子元素,用于设置一个键值对 |
Bean 的作用域
在 Spring
中可以为 Bean
指定作用域:
Bean 的实例化
在 Spring
框架中,如果想使用 Spring
容器中的 Bean
,需要对其进行实例化。Spring
框架实例化 Bean
有3种方式,即构造方法实例化、静态工厂实例化和实例工厂实例化。
(1)构造方法实例化
在 Spring
框架中,Spring
容器可以调用 Bean
对应类的无参构造方式来实例化 Bean
:
public class BookDaoImpl implements BookDao {@Overridepublic void save() {}
}
<bean id="bookDao" class="com.dao.impl.BookDaoImpl"/>
注意:若没有无参构造方法,则将抛出
BeanCreationException
异常
(2)静态工程实例化
在使用静态工厂实例化 Bean
时,开发者需要在工厂类中创建一个静态方法来创建 Bean
的实例。在配置 Bean
时,class
属性指定静态工厂类,同时还需要使用 factory-method
属性指定工厂类中的静态方法:
public class BookDaoImpl implements BookDao {@Overridepublic void save() {}
}// ...
public class BookDaoFactory {public static BookDao getBookDao() {return new BookDaoImpl();}
}
<bean id="bookDao" class="com.factory.BookDaoFactory" factory-method="getBookDao"/>
(3)实例工厂实例化
在使用实例工厂实例化 Bean
时需要我们在工厂类中创建一个实例方法来创建 Bean
的实例。在配置 Bean
时需要使用 factory-bean
属性指定配置的实例工厂,同时还需要使用 factory-method
属性指定实例工厂中的实例方法:
public class BookDaoImpl implements BookDao {@Overridepublic void save() {}
}// ...
public class BookDaoFactory {public BookDao getBookDao() {return new BookDaoImpl();}
}
<bean id="bookDaoFactory" class="com.factory.BookDaoFactory"/>
<bean id="bookDao" factory-method="getBookDao" factory-bean="bookDaoFactory"/>
除了上面的实现方式,还可以对实例工厂实例化方式进行简化,即使用 FactoryBean
实例化 Bean
。创建一个实现 FactoryBean
接口的工厂方法,实现统一接口,最后进行配置:
public class BookDaoFactoryBean implements FactoryBean<BookDao> {@Overridepublic BookDao getObject() throws Exception {return new BookDaoImpl();}@Overridepublic Class<?> getObjectType() {return BookDao.class;}@Overridepublic boolean isSingleton() {return true; // 使用单例模式}
}
<bean id="bookDao" class="com.factory.BookDaoFactoryBean"/>
Bean 的生命周期
Bean
的生命周期是指 Bean
从创建到消亡的完整过程。而 Bean
生命周期控制是指在 Bean
创建后到销毁前做一些事情。
public class BookDaoImpl implements BookDao {@Overridepublic void save() {}// 用于实现 bean 初始后对应操作public void init() {System.out.println("init...");}// 用于实现 bean 销毁前对应操作public void destroy() {System.out.println("destroy...");}
}
<bean id="bookDao" class="com.dao.impl.BookDaoImpl" init-method="init" destroy-method="destroy"/>
上面是通过配置方法的方式实现 Bean
生命周期控制,也可以通过接口的方式实现 Bean
生命周期控制,实现 InitializingBean
和 DisposableBean
接口以及对应的方法:
public class BookDaoImpl implements BookDao, InitializingBean, DisposableBean {@Overridepublic void save() {}@Overridepublic void afterPropertiesSet() throws Exception {System.out.println("init...");}@Overridepublic void destroy() throws Exception {System.out.println("destroy...");}
}
<bean id="bookDao" class="com.dao.impl.BookDaoImpl"/>
我们运行 Bean
生命周期控制的代码会发现:init
执行了,但是 destroy
没有执行。原因是:程序运行在虚拟机中,在虚拟机启动后,会经历容器初始化、使用 Bean
以及关闭/销毁容器三个阶段。然而,这里虚拟机并没有进入关闭/销毁容器阶段就退出了。这里有两种进入关闭/销毁阶段的方式:
(1)在虚拟机退出前,主动关闭容器
ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
ctx.close();
(2)注册关闭钩子
ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
ctx.registerShutdownHook(); // 指示在关闭虚拟机之前先关闭容器
Bean 的装配
Bean
的装配可以理解为将 Bean
依赖注入到 Spring
容器中。Spring
容器支持基于 XML
配置的装配、基于注解的装配以及自动装配等多种装配方式。
(1)基于 XML
配置的装配
在 Spring
中,有两种主要的基于 XML
配置的装配方式:一种是通过属性的 setter
方法进行注入,另一种是通过构造方法进行注入。根据注入的数据类型,这些方式又可以分为引用类型注入和简单类型注入。
- 使用
setter
方法注入简单类型数据
在 bean
中定义简单类型数据并提供可访问的 set
方法:
public class BookDaoImpl implements BookDao {private int connectionNum;private String databaseName;public void setConnectionNum(int connectionNum) {this.connectionNum = connectionNum;}public void setDatabaseName(String databaseName) {this.databaseName = databaseName;}
}
配置中使用 property
标签 value
属性注入简单类型数据:
<bean id="bookDao" class="com.dao.impl.BookDaoImpl"><property name="connectionNum" value="10"/><property name="databaseName" value="mysql"/>
</bean>
- 使用
setter
方法注入引用类型数据
在 bean
中定义引用类型属性并提供可访问的 set
方法:
public class BookServiceImpl implements BookService {private BookDao bookDao;public void setBookDao(BookDao bookDao) {this.bookDao = bookDao;}
}
配置中使用 property
标签 ref
属性注入引用类型:
<bean id="bookService" class="com.service.impl.BookServiceImpl"><property name="bookDao" ref="bookDao"/>
</bean>
<bean id="bookDao" class="com.dao.impl.BookDaoImpl"/>
- 使用构造方法注入简单类型数据
在 bean
中定义简单类型属性并提供可访问的构造方法:
public class BookDaoImpl implements BookDao {private int connectionNum;private String databaseName;public BookDaoImpl(int connectionNum, String databaseName) {this.connectionNum = connectionNum;this.databaseName = databaseName;}
}
配置中使用 constructor-arg
标签 value
属性注入简单类型对象:
<bean id="bookDao" class="com.dao.impl.BookDaoImpl"><constructor-arg name="connectionNum" value="10"/><constructor-arg name="databaseName" value="mysql"/>
</bean>
- 使用构造方法注入引用类型数据
在 bean
中定义引用类型属性并提供可访问的构造方法:
public class BookServiceImpl implements BookService {private BookDao bookDao;public BookServiceImpl(BookDao bookDao) {this.bookDao = bookDao;}
}
配置中使用 constructor-arg
标签 ref
属性注入引用类型对象:
<bean id="bookDao" class="com.dao.impl.BookDaoImpl"/>
<bean id="bookService" class="com.service.impl.BookServiceImpl"><constructor-arg name="bookDao" ref="bookDao"/>
</bean>
(2)基于注解的装配
使用 @Component
定义 bean
:
@Component("bookDao") // 起名称为 bookDao
public class BookDaoImpl implements BookDao {@Overridepublic void save() {}
}
核心配置文件中通过组件扫描加载 bean
:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"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.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"><context:component-scan base-package="com.dao"/>
</beans>
另外,Spring
提供了 @Component
注解的三个衍生注解,分别是:
@Controller
:用于表现层bean
定义@Service
:用于业务层bean
定义@Repository
:用于数据层bean
定义
(3)自动装配
IoC
容器根据 bean
所依赖的资源在容器中自动查找并注入到 bean
中的过程称为自动装配。自动装配的方式可分为:按类型、按名称、按构造方法和默认。
- 配置中使用
bean
标签autowire
属性设置自动装配的类型:
<bean id="bookDao" class="com.dao.impl.BookDaoImpl"/>
<bean id="bookService" class="com.service.impl.BookServiceImpl" autowire="byType"/>
使用细节
- 自动装配用于引用类型依赖注入,不能对简单类型进行操作
- 使用按类型装配时(
byType
)必须保障容器中相同类型的bean
唯一,推荐使用 - 使用按名称装配时(
byName
)必须保障容器中具有指定名称的bean
,因变量名与配置耦合,不推荐使用 - 自动装配优先级低于
setter
注入与构造器注入,同时出现时自动装配配置失效
(4)集合的注入
public class BookDaoImpl implements BookDao {private int[] array;private List<String> list;private Set<String> set;private Map<String, String> map;private Properties properties;public void setArray(int[] array) {this.array = array;}public void setList(List<String> list) {this.list = list;}public void setSet(Set<String> set) {this.set = set;}public void setMap(Map<String, String> map) {this.map = map;}public void setProperties(Properties properties) {this.properties = properties;}
}
<bean id="bookDao" class="com.dao.impl.BookDaoImpl"><property name="array"><array><value>100</value><value>200</value></array></property><property name="list"><list><value>java se</value><value>java ee</value></list></property><property name="set"><set><value>java se</value><value>java ee</value></set></property><property name="map"><map><entry key="k1" value="v1"/><entry key="k2" value="v2"/></map></property><property name="properties"><props><prop key="k1">v1</prop><prop key="k2">v2</prop></props></property>
</bean>
相关文章:

【Spring】Bean
Spring 将管理对象称为 Bean。 Spring 可以看作是一个大型工厂,用于生产和管理 Spring 容器中的 Bean。如果要使用 Spring 生产和管理 Bean,那么就需要将 Bean 配置在 Spring 的配置文件中。Spring 框架支持 XML 和 Properties 两种格式的配置文件&#…...

深入解析TK技术下视频音频不同步的成因与解决方案
随着互联网和数字视频技术的飞速发展,音视频同步问题逐渐成为网络视频播放、直播、编辑等过程中不可忽视的技术难题。尤其是在采用TK(Transmission Keying)技术进行视频传输时,由于其特殊的时序同步要求,音视频不同步现…...

为什么要使用Ansible实现Linux管理自动化?
自动化和Linux系统管理 多年来,大多数系统管理和基础架构管理都依赖于通过图形或命令行用户界面执行的手动任务。系统管理员通常使用清单、其他文档或记忆的例程来执行标准任务。 这种方法容易出错。系统管理员很容易跳过某个步骤或在某个步骤上犯错误。验证这些步…...

Android:任意层级树形控件(有效果图和Demo示例)
先上效果图: 1.创建treeview文件夹 2.treeview -> adapter -> SimpleTreeAdapter.java import android.content.Context; import android.view.View; import android.view.ViewGroup; import android.widget.ImageView; import android.widget.ListView; i…...

C++ 容器全面剖析:掌握 STL 的奥秘,从入门到高效编程
引言 C 标准模板库(STL)提供了一组功能强大的容器类,用于存储和操作数据集合。不同的容器具有独特的特性和应用场景,因此选择合适的容器对于程序的性能和代码的可读性至关重要。对于刚接触 C 的开发者来说,了解这些容…...
C++---类型转换
文章目录 C的类型转换C的4种强制类型转换RTTI C的类型转换 类型转换 内置类型之间的转换 // a、内置类型之间 // 1、隐式类型转换 整形之间/整形和浮点数之间 // 2、显示类型的转换 指针和整形、指针之间 int main() {int i 1;// 隐式类型转换double d i;printf("%d…...

CSS基础学习练习题
编程题 1.为下面这段文字定义字体样式,要求字体类型指定多种、大小为14px、粗细为粗体、颜色为蓝色。 “有规划的人生叫蓝图,没规划的人生叫拼图。” 代码: <!DOCTYPE html> <html lang"en"> <head><me…...
TypeScript知识点总结和案例使用
TypeScript 是一种由微软开发的开源编程语言,它是 JavaScript 的超集,提供了静态类型检查和其他一些增强功能。以下是一些 TypeScript 的重要知识点总结: 1. 基本类型 TypeScript 支持多种基本数据类型,包括: numbe…...
解决BUG: Since 17.0, the “attrs“ and “states“ attributes are no longer used.
从Odoo 17.0开始,attrs和states属性不再使用,取而代之的是使用depends和domain属性来控制字段的可见性和其他行为。如果您想要在选择国家之后继续选择州,并且希望在选择了国家之后才显示州字段,您可以使用depends属性来实现这一点…...

单片机GPIO中断+定时器 实现模拟串口接收
单片机GPIO中断定时器 实现模拟串口接收 解决思路代码示例 解决思路 串口波特率9600bps,每个bit约为1000000us/9600104.16us; 定时器第一次定时时间设为52us即半个bit的时间,其目的是偏移半个bit时间,之后的每104us采样并读取1bit数据。使得…...

《深入理解 Spring MVC 工作流程》
一、Spring MVC 架构概述 Spring MVC 是一个基于 Java 的轻量级 Web 应用框架,它遵循了经典的 MVC(Model-View-Controller)设计模式,将请求、响应和业务逻辑分离,从而构建出灵活可维护的 Web 应用程序。 在 Spring MV…...

HTML简介
知识点一 HTML 什么是HTML? 超文本标记语言(HyperTextMarkup Language,简称HTML) 怎么学HTML? HTML 是一门标记语言,标记语言由一套标记标签组成,学习 HTML,其实就是学习标签 开发工具 编辑器: Pycha…...

Linux系统Centos设置开机默认root用户
目录 一. 教程 二. 部分第三方工具配置也无效 一. 教程 使用 Linux 安装Centos系统的小伙伴大概都知道,我们进入系统后,通常都是自己设置的普通用户身份,而不是 root 超级管理员用户,导致我们在操作文件夹时往往爆出没有权限&am…...
【网络安全 | 甲方建设】双/多因素认证、TOTP原理及实现
未经许可,不得转载。 文章目录 背景双因素、多因素认证双因素认证(2FA)多因素认证(MFA)TOTP实现TOTP生成流程TOTP算法TOTP代码示例(JS)Google Authenticator总结背景 在传统的在线银行系统中,用户通常只需输入用户名和密码就可以访问自己的账户。然而,如果密码不慎泄…...
Nuxt3 动态路由URL不更改的前提下参数更新,NuxtLink不刷新不跳转,生命周期无响应解决方案
Nuxt3 动态路由URL不更改的前提下参数更新,NuxtLink不刷新不跳转,生命周期无响应解决方案 首先说明一点,Nuxt3 的动态路由响应机制是根据 URL 是否更改,参数的更改并不会触发 Router 去更新页面,这在 Vue3 上同样存在…...
2024华为java面经
华为2024年Java招聘面试题目可能会涵盖Java基础知识、核心技术、框架与工具、项目经验以及算法与数据结构等多个方面。以下是考的内容。 一、Java基础知识 Java中有哪些基本数据类型? Java为什么能够跨平台运行? String是基本数据类型吗?能…...
2021 年 9 月青少年软编等考 C 语言三级真题解析
目录 T1. 课程冲突思路分析T2. 余数相同问题思路分析T3. 生成括号思路分析T4. 广义格雷码思路分析T5. 菲波那契数列思路分析T1. 课程冲突 小 A 修了 n n n 门课程,第 i i i 门课程是从第 a i a_i ai 天一直上到第 b i b_i bi 天。 定义两门课程的冲突程度为:有几天…...

深度解析FastDFS:构建高效分布式文件存储的实战指南(下)
接上篇:《深度解析FastDFS:构建高效分布式文件存储的实战指南(上)》 传送门: link 文章目录 六、常用命令七、FastDFS配置详解7.1 tracker配置文件7.2 tracker目录及文件结构7.3 storage配置文件7.4 storage服务器的目录结构和文件…...

Python学习29天
二分查找 # 定义函数冒泡排序法从大到小排列 def bbble_sort(list):# i控制排序次数for i in range(len(list) - 1):# j控制每次排序比较次数for j in range(len(list) - 1 - i):if list[j] < list[j 1]:list[j], list[j 1] list[j 1], list[j] # 定义二分查找函数 def…...

Soul App创始人张璐团队携多模态大模型参加GITEX GLOBAL,展现未来社交趋势
作为中东地区规模最大、最成功的计算机通讯及消费性电子产品展,GITEX GLOBAL一直颇受全球关注,于今年迎来了第44届盛会。自诞生以来,GITEX GLOBAL始终聚焦技术驱动的创新,吸引了许多科技巨头、创新企业及投资者的参与。Soul App作为中国较早将AI技术引入社交的企业,今年首次亮相…...

工业安全零事故的智能守护者:一体化AI智能安防平台
前言: 通过AI视觉技术,为船厂提供全面的安全监控解决方案,涵盖交通违规检测、起重机轨道安全、非法入侵检测、盗窃防范、安全规范执行监控等多个方面,能够实现对应负责人反馈机制,并最终实现数据的统计报表。提升船厂…...

智慧工地云平台源码,基于微服务架构+Java+Spring Cloud +UniApp +MySql
智慧工地管理云平台系统,智慧工地全套源码,java版智慧工地源码,支持PC端、大屏端、移动端。 智慧工地聚焦建筑行业的市场需求,提供“平台网络终端”的整体解决方案,提供劳务管理、视频管理、智能监测、绿色施工、安全管…...
大模型多显卡多服务器并行计算方法与实践指南
一、分布式训练概述 大规模语言模型的训练通常需要分布式计算技术,以解决单机资源不足的问题。分布式训练主要分为两种模式: 数据并行:将数据分片到不同设备,每个设备拥有完整的模型副本 模型并行:将模型分割到不同设备,每个设备处理部分模型计算 现代大模型训练通常结合…...
大语言模型(LLM)中的KV缓存压缩与动态稀疏注意力机制设计
随着大语言模型(LLM)参数规模的增长,推理阶段的内存占用和计算复杂度成为核心挑战。传统注意力机制的计算复杂度随序列长度呈二次方增长,而KV缓存的内存消耗可能高达数十GB(例如Llama2-7B处理100K token时需50GB内存&a…...

SAP学习笔记 - 开发26 - 前端Fiori开发 OData V2 和 V4 的差异 (Deepseek整理)
上一章用到了V2 的概念,其实 Fiori当中还有 V4,咱们这一章来总结一下 V2 和 V4。 SAP学习笔记 - 开发25 - 前端Fiori开发 Remote OData Service(使用远端Odata服务),代理中间件(ui5-middleware-simpleproxy)-CSDN博客…...

人机融合智能 | “人智交互”跨学科新领域
本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...

安全突围:重塑内生安全体系:齐向东在2025年BCS大会的演讲
文章目录 前言第一部分:体系力量是突围之钥第一重困境是体系思想落地不畅。第二重困境是大小体系融合瓶颈。第三重困境是“小体系”运营梗阻。 第二部分:体系矛盾是突围之障一是数据孤岛的障碍。二是投入不足的障碍。三是新旧兼容难的障碍。 第三部分&am…...
BLEU评分:机器翻译质量评估的黄金标准
BLEU评分:机器翻译质量评估的黄金标准 1. 引言 在自然语言处理(NLP)领域,衡量一个机器翻译模型的性能至关重要。BLEU (Bilingual Evaluation Understudy) 作为一种自动化评估指标,自2002年由IBM的Kishore Papineni等人提出以来,…...

如何应对敏捷转型中的团队阻力
应对敏捷转型中的团队阻力需要明确沟通敏捷转型目的、提升团队参与感、提供充分的培训与支持、逐步推进敏捷实践、建立清晰的奖励和反馈机制。其中,明确沟通敏捷转型目的尤为关键,团队成员只有清晰理解转型背后的原因和利益,才能降低对变化的…...

tauri项目,如何在rust端读取电脑环境变量
如果想在前端通过调用来获取环境变量的值,可以通过标准的依赖: std::env::var(name).ok() 想在前端通过调用来获取,可以写一个command函数: #[tauri::command] pub fn get_env_var(name: String) -> Result<String, Stri…...