前言
欢迎来到本篇文章,鸽了好久了,今天继续写下 Spring 的内容:Spring 中 Bean 的基本概念、基本写法和 3 种实例化 Bean 的方式等。
什么是 Bean?
我们回顾下,什么是 Bean?这在上一篇文章 Spring 核心概念之一 IoC 中说过了,简而言之,一句话:被 Spring IoC 管理的对象,就是 Bean。
一个 Spring IoC 容器中管理着一个或多个 Bean,这些 Bean 是由我们提供给容器的配置元数据创建的(比如以 XML <bean /> 形式定义的 Bean)。
「Bean Definition」的属性
在容器本身中,这些 Bean 定义被表示为 BeanDefinition 对象,它包含(除其他信息外)以下元数据(metadata)。
- 一个全路径类名:比如 cn.god23bin.demo.controller.DemoController,这就是 Bean Definition 的实际实现类。
- Bean 的行为配置元素:它说明了 Bean 在容器中的行为方式(比如 scope、生命周期回调等等)。
- 对其他 Bean 的引用:这些被引用的 Bean 被称为协作者或依赖(collaborators or dependencies)。
- 其他配置的设置:比如,数据库连接池的大小限制或使用的连接数。
这些元数据对应着每个 Bean Definition 的一组属性。下表描述了这些属性:
属性 | 解释… |
Class | 该属性是必需的,它指定了用于创建 bean 的 bean 类。 |
Name | 该属性唯一地指定 bean 标识符。 在基于 XML 的配置元数据中,我们可以使用 id 或 name 属性来指定 bean 标识符。 |
Scope | 该属性指定从特定 bean 定义创建的对象的范围,将在 bean 范围一章中讨论。 |
Constructor arguments | 这用于注入依赖关系,将在后续章节中讨论。 |
Properties | 这用于注入依赖关系,将在后续章节中讨论。 |
Autowiring mode | 这用于注入依赖关系,将在后续章节中讨论。 |
Lazy initialization mode | 懒加载模式,让 bean 告诉 IoC 容器在第一次被请求时创建一个 bean 实例,而不是在启动时。 |
Initialization method | 在容器设置了 bean 上的所有必要属性之后调用的回调。 它将在 bean 生命周期章节中讨论。 |
Destruction method | 当包含 bean 的容器被销毁时要使用的回调。 它将在 bean 生命周期章节中讨论。 |
Bean 的命名
以 XML 作为配置元数据的方式中,Bean 定义的基本格式是这样写的:
xml复制代码<?xml version="1.0" encoding="UTF-8"?>
<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
https://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="..." class="...">
<!-- 该 bean 的协作者和配置写在此处 -->
</bean>
<bean id="..." class="...">
<!-- 该 bean 的协作者和配置写在此处 -->
</bean>
<!-- 更多的 bean 定义写在此处 -->
</beans>
XML 中类似 <bean/> 由尖括号组成的写法,一般称为「标签」或者「元素」,在后面的说法中,可能有时出现标签的说法或者元素的说法,实际上都是指同一个东西,后续便不再说明
我们可以看到,bean 标签有 id 、class 属性。
这个 id 是每个 Bean 都有的一个标识符(identifier),这些标识符在 Spring IoC 容器中必须是唯一的。
一个 Bean 一般只有一个标识符。如果它需要一个以上的标识符,那么多余的标识符可以被视为别名。
在基于 XML 的配置元数据中,我们可以使用 id 属性、name 属性或两者来指定 Bean 的标识符。
命名习惯
习惯上,这里对于 Bean 定义的命名和对于我们命名 Java 类中的属性名是一样的,就是驼峰命名,小写字母开头。比如 userService、userDao、loginController 等等。
还有,如果想起多个别名,那么用 name 属性指定别名是什么,比如 name="别名1,别名2,别名3",多个别名之间可以用英文逗号、英文分号或者空格分隔。
Bean 的别名
除了使用 Bean 定义中的 name 属性来对 Bean 起别名,还可以使用 alias 标签给 Bean 起别名:
xml复制代码<bean id = "userService" name="userServiceName" ... ></bean>
<alias name="userServiceName" alias="userServiceAliasName" />
这样,名为 userServiceName 的 Bean 也有了另一个名字:userServiceAliasName。
Spring 实例化 Bean 的 3 种方式
Spring 可以通过 3 种方式来为我们创建 Bean 对象,创建的对象是根据我们定义的配置元数据来进行创建的。
这 3 种方式分别是:
- 通过构造方法实例化
- 通过静态工厂方法实例化
- 通过实例工厂方法实例化
在上面的 Bean 定义的属性中,我们也看到了,在 <bean /> 中有一个 class 属性,就是用来指定要实例化的对象的类型的。一般情况下,这个 class 属性是必须写的,除非是通过实例工厂方法实例化的 Bean,那么它的 Bean 定义可以不需要 class 属性。
1. 通过构造方法实例化
以 XML 为配置元数据为例,在默认的情况下,我们定义的 Bean 都是以构造方法来实例化 Bean 对象的。
这些交给 Spring 管理的 Bean,不需要实现 Spring 的某某接口或者继承 Spring 的某某类,只需要有一个无参的构造方法,就可以了。
这里也是 Spring 低侵入的特点,我们的类根本不需要去实现 Spring 特定的接口或继承特定的类进而实现 IoC 的功能。
xml复制代码<bean id="fans" class="cn.god23bin.demo.domain.entity.Fans">
<property name="name" value="god23bin" />
</bean>
以上面这个例子来说,这样定义的 Bean 就是通过
cn.god23bin.demo.domain.entity.Fans 的构造方法实例化的:
java复制代码package cn.god23bin.demo.domain.entity;
public class Fans {
private String name;
public Fans() {
System.out.println("粉丝无参构造方法被调用了!");
}
// 省略 getter 和 setter
}
从 Spring IoC 容器中获取被管理的 Fans 对象,这个过程,Spring 就会根据配置元数据去使用构造方法实例化 Fans 对象。
测试:
java复制代码ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
Fans fans = applicationContext.getBean(Fans.class);
System.out.println("fans name: " + fans.getName());
输出:
console复制代码粉丝无参构造方法被调用了!
fans name: god23bin
2. 通过静态工厂方法实例化
假设我们有一些对象是从静态工厂中获取对象的,有一个静态工厂类,类中定义了一个静态方法,该方法是创建对象的:
java复制代码package cn.god23bin.demo.domain.entity.factory;
import cn.god23bin.demo.domain.entity.Fans;
public class FansStaticFactory {
// 返回 Fans 对象的静态方法
public static Fans createFansInstance() {
System.out.println("粉丝静态工厂!");
Fans fans = new Fans();
fans.setName("练习两年半 | 你干嘛哎哟~~");
return fans;
}
}
接着在配置元数据中定义 Bean,将通过调用工厂的静态方法来创建 Bean 对象,使用 class 属性指定包含了静态工厂方法的类,使用 factory-method 属性指定创建对象的静态方法:
xml复制代码<bean id="fans" class="cn.god23bin.demo.domain.entity.factory.FansStaticFactory" factory-method="createFansInstance" />
测试:
java复制代码ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
Fans fans = applicationContext.getBean(Fans.class);
System.out.println("fans name: " + fans.getName());
输出:
console复制代码粉丝静态工厂!
粉丝无参构造方法被调用了!
fans name: 练习两年半 | 你干嘛哎哟~~
3. 通过实例工厂方法实例化
实例工厂:
java复制代码package cn.god23bin.demo.domain.entity.factory;
import cn.god23bin.demo.domain.entity.Fans;
public class FansFactory {
public FansFactory() {
}
public Fans createFansInstance() {
System.out.println("粉丝工厂");
Fans fans = new Fans();
fans.setName("练习两年半 | 你干嘛哎哟~~ | 记得关注我噢~");
return fans;
}
}
和静态工厂实例化 Bean 对象类似,我们这里用实例厂方法进行 Bean 的实例化,从 Spring IoC 容器中调用现有工厂 Bean 的非静态方法来创建一个新的 Bean。
这里需要注意 Fans 这个 Bean 的 class 属性是不需要的。
xml复制代码<!-- 配置实例工厂 Bean,该对象包含了创建 Fans 对象的名为 createFansInstance 的方法 -->
<bean id="fansFactory" class="cn.god23bin.demo.domain.entity.factory.FansFactory">
<!-- 工厂的其他需要的依赖对象就配置在这里 -->
</bean>
<!-- 这个 Bean 将通过实例工厂创建 -->
<bean id="fans" factory-bean="fansFactory" factory-method="createFansInstance" />
总结
以上,就是本文的所有内容,主要介绍了 Spring 中 Bean 的概念和 Bean 的定义属性。
Bean 由配置元数据创建,比如以XML形式定义的 Bean。Bean 在容器中由 BeanDefinition 对象表示,它包含类名、行为配置元素、对其他 Bean 的引用以及其他配置设置等元数据属性。
Bean 的命名方式一般就是驼峰命名的。在基于XML的配置元数据中,可以使用 id 属性或 name 属性来指定 Bean 的标识符,Bean 的别名可以使用 alias 标签进行定义。
最后,我们说了 Spring 实例化 Bean 的三种方式:通过构造方法实例化、通过静态工厂方法实例化和通过实例工厂方法实例化。
当然,Bean 的内容不止这些,本文是对 Bean 的初步介绍。下一篇我们将介绍 Bean 之间的依赖关系,通过依赖注入来实现 Bean 之间的相互依赖。