家族传承:Spring MVC中父子容器的搭建与管理指南

父子容器

在 Spring 框架中,Spring 容器可以划分为父子容器,这个概念是为了更好地管理 Spring 的 Bean 实例而引入的。父子容器之间可以定义层次关系,其中子容器可以访问父容器中所有的 Bean,而父容器则无法访问子容器中的 Bean。

通过父子容器的划分,我们可以将应用中的不同模块进行分离,使得应用更加模块化和可管理。同时,子容器也可以访问父容器中的Bean实例,这样可以避免重复创建对象,提高应用的性能和效率。

在 SpringMVC 中,DispatcherServlet 创建的容器是子容器,而 ContextLoaderListener 创建的容器是父容器。

具体来说,ContextLoaderListener 是一个监听器,它会在 Web 应用启动时被加载,并创建一个根级别的 ApplicationContext 作为 Spring 的父容器。这个父容器负责管理整个 Web 应用的全局资源,如 Service、Repository 等。它通常会加载应用的配置文件(如 applicationContext.xml)并初始化相应的 Bean。

而 DispatcherServlet 则是一个 Servlet,它也会在 Web 应用启动时被加载,并创建一个与之关联的子容器,也就是 WebApplicationContext。这个子容器负责管理与 SpringMVC 相关的 Bean,如 Controller、ViewResolver 等。它会继承父容器的 ApplicationContext,并且可以访问父容器中的 Bean。

在这种父子容器的设计中,父容器负责管理全局资源,而子容器专注于处理与 SpringMVC 相关的逻辑。通过这种方式,可以更好地实现模块化和可管理性,同时也能够避免重复创建对象,提高应用的性能和效率。

web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0">


    <!-- 配置父工厂相关 -->
    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>

    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:spring/application-context.xml</param-value>
    </context-param>

    <!-- 配置子工厂相关 -->
    <servlet>
        <servlet-name>dispatcherServlet</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <!-- 指定 Spring MVC 配置文件位置 -->
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:spring/spring-mvc.xml</param-value>
        </init-param>
        <!-- 设置此 Servlet 在 Tomcat 启动的时候创建 -->
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>dispatcherServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

</web-app>

Application-context.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"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">

    <!--
        父工厂、父容器:管理除了 MVC 相关的其它 Bean,解决父子工厂耦合问题。
        连接池、Dao、Service、事务、MQ、Redis 等。
    -->

    <!-- 配置注解扫描路径,因为已经有了子工厂管理,所以父工厂就没必要再重复管理了,浪费空间 -->
    <context:component-scan base-package="world.xuewei">
        <context:exclude-filter type="aspectj" expression="world.xuewei.controller.*"/>
    </context:component-scan>

</beans>

Spring-mvc.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"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">

    <!--
        父工厂、父容器:管理除了 MVC 相关的其它 Bean,解决父子工厂耦合问题。
        连接池、Dao、Service、事务、MQ、Redis 等。
    -->

    <!-- 配置注解扫描路径,因为已经有了子工厂管理,所以父工厂就没必要再重复管理了,浪费空间 -->
    <context:component-scan base-package="world.xuewei">
        <context:exclude-filter type="aspectj" expression="world.xuewei.controller.*"/>
    </context:component-scan>

</beans>