• Bean 概念
  • 元数据 BeanDefinition
  • IoC 容器创建 Bean
  • XML 配置文件方式
  • 注解配置方式
  • Java 类配置方式
  • Bean 注入方式
  • 注解注入
  • Bean 自动装配
  • 参考文章

Bean 概念

Spring Framework 阅读版本为 5.2.x

由 IoC 容器管理的那些组成你应用程序的对象我们就叫它 Bean。

其实意思就是,每个我们加上注解交给 Spring 的类,都叫做 Bean。

Bean 由 Spring容器初始化、装配及管理的对象,除此之外,bean 就与应用程序中的其他对象没有什么区别了。

java 创建bean所占内存大小_spring



元数据 BeanDefinition

Bean 是由用容器提供的配置元数据 BeanDefinition 创建的。

BeanDefinition 继承了 BeanMetadataElementAttributeAccessor 接口,对于他们的作用嘛,我也是看别人说的,就贴一下过去略过:

  • BeanMetadataElement :bean 元数据,读取配置资源的能力。
  • AttributeAccessor :Spring 定义的属性访问器,对 Bean 的属性进行操作的 API;例如设置属性、获取属性、判断是否存在该属性,返回 bean 所有的属性名称等。

对于 IoC 容器而言,它需要完全掌握 bean ,首先需要了解:

  • 如何创建一个 bean
  • bean 的生命周期的详细信息
  • bean 的依赖关系

而这些具体的信息,都由元数据来定义。

BeanDefinition 只是一个接口,实际产生作用的实现类有很多,比如下面的继承关系图:

java 创建bean所占内存大小_java_02

首先,先介绍下 BeanDefinition ,源码是看不懂的,这辈子都不可能了;只能大致介绍下它的功能,这个接口的设计包含了一个 Bean 最基础的特征,AbstractBeanDefinition 包含以下字段,以下列了一些个人认为重要的;

具体可以参考:

属性

描述

beanClass

这个属性是强制性的,并且指定用来创建 bean 的 bean 类。

name

类名,这个属性指定唯一的 bean 标识符。在基于 XML 的配置元数据中,你可以使用 ID 和/或 name 属性来指定 bean 标识符。

scope

这个属性指定由特定的 bean 定义创建的对象的作用域

lazyInit

是否延迟加载,对应 bean 属性lazy-init;

延迟初始化的 bean 告诉 IoC 容器在它第一次被请求时,而不是在启动时去创建一个 bean 实例。

autowiring mode

自动注入模式,它是用来注入依赖关系的

dependsOn

用来表示一个 bean 的实例化依赖另一个 bean 先实例化,对应 bean 属性 depend-on

primary

自动装配时出现多个bean候选者时,将作为首选者,对应bean属性primary

constructorArgumentValues

记录构造函数注入属性,对应bean属性constructor-arg

propertyValues

普通属性集合

initMethodName

初始化方法,对应 bean 属性 init-method

destroyMethodName

销毁方法,对应bean属性destroy-method

所以只需要拿到 BeanDefinition, Spring IoC 容器就可以根据这些信息反射创建对象。

java 创建bean所占内存大小_ioc、aop_03

然后它的几个实现类分别具有不同的分工:

  1. AbstractBeanDefinition :它是基础抽象类,实现了 BeanDefinition 的基本功能,使得其他实现类不需要从头实现
  2. GenericBeanDefinition :通用的bean实现,自2.5以后新加入的bean文件配置属性定义类,是ChildBeanDefinitionRootBeanDefinition 更好的替代者
  3. ScannedGenericBeanDefinition : 被包扫描到的 bean 定义,@Component 注解生成
  4. AnnotatedGenericBeanDefinition : 查找类注解初始化的定义,@Configuration 注解生成
  5. RootBeanDefinition :代表一个从配置源( XMLJava Config 等)中生成的 BeanDefinition
  6. ChildBeanDefinition :可以从父 BeanDefinition 中集成构造方法,属性等

它们发挥作用的位置是在 IoC 容器中,使用 Map 存储 Bean 时,例如 DefaultListableBeanFactory

/** bean 的名字为键,BeanDefinition为值,初始容量为256 */
    private final Map<String, BeanDefinition> beanDefinitionMap = 
        new ConcurrentHashMap<String, BeanDefinition>(256);

至于更详细的工作原理,以后有兴趣了再去研究,先往下学重要的。这里简单提几个关键词:

  1. BeanDefinitionRegistryPostProcessor 接口
  2. DefaultListableBeanFactory
  3. PostProcessorRegistrationDelegate 类的 invokeBeanFactoryPostProcessors 方法



IoC 容器创建 Bean

上面图片中已经画出来了,Bean 配置信息的三种方式:

  • XML 配置文件方式
  • 注解配置方式
  • Java 类配置方式

下面来主要讲这三个方式。



XML 配置文件方式

XML 配置方式现在已经不推荐使用了,不过可能一些老项目维护还需要,把使用方式介绍下。

在 Spring 配置文件中,使用 bean 标签,标签里添加对应属性,就可以实现对象创建。

bean 中有很多属性,基本都是和上面 AbstractBeanDefinition 列的对应,稍微列几个常用的:

  • id :唯一标识,ApplicationContext.getBean 获取时的 bean 参数
  • class :创建对象类的全路径
  • name :名称属性,和 id 属性类似,不过 id 不能加特殊符号,name 可以;比较早期的属性,基本没人用了
  • property :属性参数注入,name 中加属性名
  • scope :作用域,常用的有两种:singleton 表示单例,prototype 表示多实例,默认为单例 singleton 。用的比较少的,并且只支付 WebApplicationContext 环境的: request (每次HTTP请求都会创建一个新的Bean)、session (同一个HTTP Session共享一个Bean,不同Session使用不同的Bean)、global-session (一般用于Portlet应用环境)

下面继续。。。。



  1. 在绝大部分情况下,我们使用最基本的配置就可以满足需求,如下:
<bean id="user" class="cn.shiva.demo1.User"/>

这个方式使用的是无参构造器进行 Bean 的实例化,如果该类不存在无参构造器,则会发生异常。

然后是有参构造的话,假设有个 String s 参数,一定要先创建构造方法:

<!-- 构造器方式 -->
<bean id="user" class="cn.shiva.demo1.User">
    <constructor-arg name="s" value="bean property s属性"/>
</bean>

然后就是通过 set 方式注入:

<!-- 属性注入方式 -->
<bean id="user" class="cn.shiva.demo1.User">
    <property name="s" value="bean property s属性" />
</bean>



  1. 使用工厂模式配合 XML 配置
public class UserFactory {
      // 静态方法,返回User对象
      public static User getUser() {
          return new User();
      }
  }

使用 xml 配置文件 bean 属性:

<bean id="user" class="cn.shiva.demo1.UserFactory" factory-method="getUser"/>

然后注入的其他类型,外联对象、数组、Map、List 什么的都差不多。



注解配置方式

注解配置可以简化 xml 配置。

Spring 针对 Bean 创建对象提供了以下几种注解:

  • @Component :表示是一个普通的 Bean 组件
  • @Service :作用于业务逻辑层
  • @Controller :作用于控制层,spring-mvc 的注解,进行前端请求的处理,转发,重定向
  • @Repository :作用于持久层,作为DAO对象(数据访问对象,Data Access Objects),这些类可以直接对数据库进行操作

四个注解都用来创建 Bean 实例,Component 是基础注解,其他三个都是继承自它,然后再扩展功能。



Java 类配置方式

最后就是使用 @Configuration 的 Java 配置方式了,

通过 @Configuration 注解来表明该类是一个 Spring 的配置;

其实就是一个简化的 xml 配置文件

@Configuration
@ComponentScan(basePackages = "cn.shiva.demo3")
public class SpringConfig {
    // 通过@Bean注解来表明是一个Bean对象,相当于xml中的<bean>
    @Bean
    public UserDAO getUserDAO() {
        return new UserDAO(); // 直接new对象做演示
    }
}



Bean 注入方式

Bean 注入方式其实和创建方式类似。

XML 注入方式也有构造函数注入、setter 方法注入、工厂注入;这些就不讲了,现在 SpringBoot 已经没见过这种玩意儿了。



注解注入

这几种注解,我估计所有人都已经熟得不能再熟了:

  • @AutoWired :根据属性的类型 byType 自动注入,基础用法;
  • @Qualifier :根据属性的名称 byName 自动注入;

使用这个注解的时候,需要和 AutoWired 一起使用。

如果一个 UserService ,存在两个实现类 UserServiceImpl1UserServiceImpl2 ,就需要根据名称来进行选择注入;

使用示例:

@AutoWired
@Qualifier(value = "userServiceImpl2")
private UserService userService;
  • @Resource :可以根据类型注入,也可以根据名称注入;默认使用属性的名称

Resource 是 Javax 的注解,Spring 官方肯定建议我们使用它自己的注解,但是功能是可以实现的;用法相同。

  • @Value :注入普通类型属性,我个人用得比较多的是在 SpringBoot 中读取配置文件的自定义配置


Bean 自动装配

Spring 提供了四种自动装配的类型,在接口 AutowireCapableBeanFactory 进行了列举:

  • no: 显式指明不使用 Spring 的自动装配功能
  • byName:根据属性和组件的名称匹配关系来实现bean的自动装配
  • byType:根据属性和组件的类型匹配关系来实现bean的自动装配,有多个适合类型的对象时装配失败
  • constructor:与 byType 类似是根据类型进行自动装配,但是要求待装配的 bean 有相应的构造函数

一般来说,我们接触到的就 byName 根据属性的名称 和 byType 根据属性的类型两种装配方式。

自动装配的原理以后再讲,Spring 的源码实在是。。。看不懂。