## ClassPathXmlApplicationContext - [ClassPathXmlApplicationContext](#classpathxmlapplicationcontext) - [一、基本信息](#一基本信息) - [二、知识储备](#二知识储备) - [三、基本描述](#三基本描述) - [四、主要功能](#四主要功能) - [五、最佳实践](#五最佳实践) - [六、时序图](#六时序图) - [七、源码分析](#七源码分析) ### 一、基本信息 ✒️ **作者** - Lex 📝 **博客** - [掘金](https://juejin.cn/user/4251135018533068/posts) 📚 **源码地址** - [github](https://github.com/xuchengsheng/spring-reading) ### 二、知识储备 1. **XmlBeanDefinitionReader** + [XmlBeanDefinitionReader](https://github.com/xuchengsheng/spring-reading/blob/master/spring-beans/spring-bean-xmlBeanDefinitionReader/README.md)是Spring Framework中的一个类,用于加载和解析XML格式的Bean定义配置文件,将配置文件中定义的Bean元数据信息提取为Spring容器内部的Bean定义对象,进而实现IOC容器的构建和管理。这类负责读取XML配置文件,解析Bean的定义信息(包括ID、类名、属性、依赖等),并将这些定义注册到Spring应用程序上下文,使我们能够方便地配置和管理应用程序中的各种Bean组件。 ### 三、基本描述 `ClassPathXmlApplicationContext` 是 Spring 框架中用于从类路径(classpath)加载 XML 配置文件并初始化 Spring 容器的一种方式。它是 `ApplicationContext` 接口的实现类之一,负责读取配置文件,解析配置信息,然后创建和管理 Spring 容器中的 bean 实例。 ### 四、主要功能 1. **加载配置文件** + 主要功能是加载指定的 XML 配置文件,该配置文件包含了应用程序中各个组件(bean)的定义、依赖关系、配置信息等。 2. **容器初始化** + `ClassPathXmlApplicationContext` 在被实例化时,会读取并解析配置文件,然后初始化 Spring 容器。这个过程包括创建和管理 bean 实例、解决 bean 之间的依赖关系等。 3. **获取 bean 实例** + 通过容器的 `getBean` 方法,可以从容器中获取在配置文件中定义的 bean 实例。 4. **IoC(控制反转)** + `ClassPathXmlApplicationContext` 是 IoC 容器的一种实现,它负责管理和控制组件的生命周期。在容器初始化时,会根据配置文件中的信息实例化和装配 bean,而不是由应用程序代码直接创建对象。 5. **依赖注入** + 容器通过读取配置文件中的信息,自动解决 bean 之间的依赖关系。这意味着在配置文件中声明的 bean 可以通过属性注入或构造函数注入的方式获取其依赖的其他 bean。 6. **AOP(面向切面编程)** + `ClassPathXmlApplicationContext` 支持通过配置文件定义切面和通知,实现横切关注点的分离,使得应用程序的关注点更加清晰和模块化。 7. **事件传播** + Spring 容器支持事件机制,`ClassPathXmlApplicationContext` 可以发布应用程序中发生的事件,以便其他组件能够监听并作出相应的响应。 ### 五、最佳实践 通过 `ClassPathXmlApplicationContext` 构造方法创建 Spring 容器的实例,加载类路径下的 "beans.xml" 配置文件。使用容器的 `getBean` 方法,通过指定 bean 的类型(`MyBean.class`)获取在配置文件中定义的 bean 实例,并将其打印出来。 ```java public class ClassPathXmlApplicationContextDemo { public static void main(String[] args) { ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("classpath:beans.xml"); System.out.println("MyBean = " + context.getBean(MyBean.class)); } } ``` 加载类路径下名为 "`classpath:beans.xml`" 的资源文件的内容。在我们的示例配置文件中,这个资源文件定义了一个名为 "`myBean`" 的 Spring Bean,该 Bean 具有一个属性 "message",其值设置为 "Hello World"。 ```xml ``` `MyBean` 的Java类,代表了一个简单的Java Bean。 ```java public class MyBean { private String message; public String getMessage() { return message; } public void setMessage(String message) { this.message = message; } } ``` ### 六、时序图 ~~~mermaid sequenceDiagram Title: ClassPathXmlApplicationContext时序图 ClassPathXmlApplicationContextDemo->>ClassPathXmlApplicationContext:ClassPathXmlApplicationContext(configLocation) Note over ClassPathXmlApplicationContextDemo,ClassPathXmlApplicationContext: 创建ClassPathXmlApplicationContext实例 ClassPathXmlApplicationContext->>ClassPathXmlApplicationContext:ClassPathXmlApplicationContext(configLocations,refresh,parent) Note over ClassPathXmlApplicationContext: 通过多个配置文件路径、刷新标志和父上下文构造实例 ClassPathXmlApplicationContext->>AbstractApplicationContext:refresh() Note over ClassPathXmlApplicationContext,AbstractApplicationContext: 刷新应用程序上下文 AbstractApplicationContext->>AbstractApplicationContext:obtainFreshBeanFactory() Note over AbstractApplicationContext: 获取刷新过的bean工厂 AbstractApplicationContext->>AbstractRefreshableApplicationContext:refreshBeanFactory() Note over AbstractApplicationContext,AbstractRefreshableApplicationContext: 调用子类的refreshBeanFactory() AbstractRefreshableApplicationContext->>AbstractXmlApplicationContext:loadBeanDefinitions(beanFactory) Note over AbstractRefreshableApplicationContext,AbstractXmlApplicationContext: 调用子类的loadBeanDefinitions方法 AbstractXmlApplicationContext->>XmlBeanDefinitionReader:new XmlBeanDefinitionReader(beanFactory) Note over AbstractXmlApplicationContext,XmlBeanDefinitionReader: 创建XmlBeanDefinitionReader实例 XmlBeanDefinitionReader->>AbstractXmlApplicationContext:返回Bean定义读取器 Note over XmlBeanDefinitionReader,AbstractXmlApplicationContext: 返回Bean定义读取器 AbstractXmlApplicationContext->>AbstractXmlApplicationContext:loadBeanDefinitions(reader) Note over AbstractXmlApplicationContext: 调用XmlBeanDefinitionReader的loadBeanDefinitions方法 AbstractXmlApplicationContext->>XmlBeanDefinitionReader:reader.loadBeanDefinitions(configLocations) Note over AbstractXmlApplicationContext,XmlBeanDefinitionReader: 加载配置文件中的Bean定义 AbstractApplicationContext->>AbstractRefreshableApplicationContext:getBeanFactory() Note over AbstractApplicationContext,AbstractRefreshableApplicationContext: 获取Bean工厂 AbstractRefreshableApplicationContext->>AbstractApplicationContext:返回Bean工厂 Note over AbstractApplicationContext: 返回Bean工厂 ~~~ ### 七、源码分析 在`org.springframework.context.support.ClassPathXmlApplicationContext#ClassPathXmlApplicationContext(configLocation)`方法中,又调用了另外一个构造方法。 ```java public ClassPathXmlApplicationContext(String configLocation) throws BeansException { this(new String[] {configLocation}, true, null); } ``` 在`org.springframework.context.support.ClassPathXmlApplicationContext#ClassPathXmlApplicationContext(configLocations, refresh,parent)`方法中,首先设置了父应用程序上下文和配置文件位置,然后根据是否需要刷新,决定是否立即执行应用程序上下文的刷新操作。 ```java public ClassPathXmlApplicationContext( String[] configLocations, boolean refresh, @Nullable ApplicationContext parent) throws BeansException { super(parent); setConfigLocations(configLocations); if (refresh) { refresh(); } } ``` 在`org.springframework.context.support.AbstractApplicationContext#refresh`方法中,调用 `obtainFreshBeanFactory` 方法来刷新内部的 bean 工厂,从而触发容器的初始化过程。 ```java @Override public void refresh() throws BeansException, IllegalStateException { // ... [代码部分省略以简化] // 该方法由子类实现,用于刷新内部的 bean 工厂。 ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory(); // ... [代码部分省略以简化] } ``` 在`org.springframework.context.support.AbstractApplicationContext#obtainFreshBeanFactory`方法中,首先调用 `refreshBeanFactory` 方法来刷新内部的 bean 工厂,并最终返回这个刷新过的 bean 工厂。 ```java protected ConfigurableListableBeanFactory obtainFreshBeanFactory() { refreshBeanFactory(); return getBeanFactory(); } ``` 在`org.springframework.context.support.AbstractRefreshableApplicationContext#refreshBeanFactory`方法中,首先检查是否已存在 bean 工厂,如果存在则先销毁已有的 bean 并关闭 bean 工厂。接着,创建一个新的 `DefaultListableBeanFactory` 实例,设置其序列化 ID 为容器的 ID,然后调用 `customizeBeanFactory` 定制 bean 工厂。随后,通过 `loadBeanDefinitions` 加载 bean 的定义,包括读取配置文件、解析 bean 的定义等。 ```java @Override protected final void refreshBeanFactory() throws BeansException { // 如果已经存在 bean 工厂,则销毁已有的 bean 并关闭 bean 工厂 if (hasBeanFactory()) { destroyBeans(); closeBeanFactory(); } try { // 创建一个新的 DefaultListableBeanFactory 实例 DefaultListableBeanFactory beanFactory = createBeanFactory(); // 设置 bean 工厂的序列化 ID 为容器的 ID beanFactory.setSerializationId(getId()); // 定制 bean 工厂,由子类实现 customizeBeanFactory(beanFactory); // 加载 bean 的定义,包括读取配置文件、解析 bean 的定义等 loadBeanDefinitions(beanFactory); // 将创建好的 bean 工厂赋值给容器 this.beanFactory = beanFactory; } catch (IOException ex) { // 如果在解析 bean 定义时发生 I/O 错误,则抛出 ApplicationContextException 异常 throw new ApplicationContextException("I/O error parsing bean definition source for " + getDisplayName(), ex); } } ``` 在`org.springframework.context.support.AbstractXmlApplicationContext#loadBeanDefinitions(beanFactory)`方法中,首先创建了一个 `XmlBeanDefinitionReader` 对象,用于读取 XML 格式的 bean 定义。然后,配置了 bean 定义阅读器的环境、资源加载器和实体解析器等属性。接着,允许子类通过 `initBeanDefinitionReader` 提供自定义的初始化操作。最后,调用阅读器的加载方法,实际上读取配置文件,解析 bean 定义,将它们注册到 bean 工厂中。 ```java protected void loadBeanDefinitions(DefaultListableBeanFactory beanFactory) throws BeansException, IOException { // 为给定的 BeanFactory 创建一个新的 XmlBeanDefinitionReader XmlBeanDefinitionReader beanDefinitionReader = new XmlBeanDefinitionReader(beanFactory); // 使用该上下文的资源加载环境配置 bean 定义阅读器 beanDefinitionReader.setEnvironment(this.getEnvironment()); // 将资源加载器设置为当前应用程序上下文 beanDefinitionReader.setResourceLoader(this); // 设置实体解析器为 ResourceEntityResolver,用于解析 bean 定义中的实体引用 beanDefinitionReader.setEntityResolver(new ResourceEntityResolver(this)); // 允许子类提供对阅读器的自定义初始化,然后继续实际加载 bean 定义 initBeanDefinitionReader(beanDefinitionReader); // 调用阅读器的加载方法,实际上会读取配置文件,解析 bean 定义,将它们注册到 bean 工厂中 loadBeanDefinitions(beanDefinitionReader); } ``` 在`org.springframework.context.support.AbstractXmlApplicationContext#loadBeanDefinitions(reader)`方法中,首先获取配置资源(`Resource`)数组,这些资源通过类路径方式加载的配置文件。如果配置资源不为空,则通过 `XmlBeanDefinitionReader` 对象加载这些资源中的 bean 定义。接着,获取配置文件路径数组,这些路径通过字符串形式指定的配置文件路径。如果配置文件路径不为空,则同样通过`XmlBeanDefinitionReader`来加载这些路径中的 bean 定义。 > **关于`reader.loadBeanDefinitions`方法的源码分析已经在另外一篇关于[XmlBeanDefinitionReader](https://github.com/xuchengsheng/spring-reading/blob/master/spring-beans/spring-bean-xmlBeanDefinitionReader/README.md)类的博客中详细分析了。** ```java protected void loadBeanDefinitions(XmlBeanDefinitionReader reader) throws BeansException, IOException { // 获取配置资源(Resource)数组,通常表示通过类路径或其他方式加载的配置文件 Resource[] configResources = getConfigResources(); // 如果配置资源不为空,则通过阅读器加载这些资源中的 bean 定义 if (configResources != null) { reader.loadBeanDefinitions(configResources); } // 获取配置文件路径数组,通常表示通过字符串形式指定的配置文件路径 String[] configLocations = getConfigLocations(); // 如果配置文件路径不为空,则通过阅读器加载这些路径中的 bean 定义 if (configLocations != null) { reader.loadBeanDefinitions(configLocations); } } ```