Spring Bean定义的解析
前言
Spring容器有两种定义方式,一种是使用XML文件定义,另外一种是Spring 2.5引入的java注解定义。两种容器定义的方式各有优缺点,由于SpringBoot使用的是注解形式的容器,所以近几年使用比较多的Spring容器定义是注解定义。本文将从这两种容器定义的实现方式、不同之处、优缺点等多方面介绍两种容器。下图为Spring容器定义的类型示例:
对于不同形式定义的Spring容器,Spring有不同类型的容器名称,下图中列举了几种常见的Spring容器:
XML容器和注解容器实现的区别
对于XML形式的Spring容器和注解形式的Spring容器,他们的主要区别其实只是读取、解析以及注册Bean定义的方式和时机不相同,当容器的定义成功初始化BeanFactory之后,二者之间就基本没区别了,两种容器有区别的主要地方如下图中红色的方框内容所示:
Spring容器重复Refresh
Spring容器可以分为两大类:GenericApplicationContext(XML)和AbstractRefreshableApplicationContext(注解),其余的容器都是这两个容器的子类,其中所有的XML容器都是AbstractRefreshableApplicationContext的子类,这些容器都有一个共同的属性:可以重复Refresh BeanFactory。而GenericApplicationContext类型的容器不允许重复Refresh。重复Refresh意味着程序在运行中可以动态的加载XML文件,然后根据新的XML文件动态生成新容器。
Refresh是Spring根据容器定义构建Spring容器的阶段,可以重复Refresh容器意味着在容器启动之后,运行再次修改容器定义,然后根据新的定义生成新的容器。以下为AbstractRefreshableApplicationContext容器多次Refresh的源码,可以看到,容器每次Refresh会做以下几件事:
- 销毁所有的单例Bean;
- 关闭已有的BeanFactory;
- 创建新的BeanFactory;
- 加载容器的定义。
/**
* This implementation performs an actual refresh of this context's underlying
* bean factory, shutting down the previous bean factory (if any) and
* initializing a fresh bean factory for the next phase of the context's lifecycle.
*/