【问题标题】:SpringMVC app... why are app contexts being initialized twice?SpringMVC 应用程序...为什么应用程序上下文被初始化两次?
【发布时间】:2013-11-18 16:15:39
【问题描述】:

我之前在这些论坛上看到过这个基本问题,但似乎没有一个答案能解决我的特定问题。无论如何,我有一个小型 Spring webapp,由核心 Spring 业务层、SpringMVC 和 Spring-Quartz 组成(我也在使用 MyBatis,尽管我认为这不相关)。我所有的 Spring 库都是 3.1.3

问题是当我将我的应用程序部署到 Tomcat 6 时,通常根应用程序上下文和 Web 应用程序上下文都被初始化两次。通过查看日志可以明显看出这一点,并且我的 Quartz 作业在应该触发一次时触发了两次这一事实也很明显(后一点就是为什么这不仅仅是一个理论问题)。

我以为我的应用程序上下文已经全部整理好,但显然我缺少一些东西。我的一般做法是这样的:

  1. 将基本的 Spring 配置放在 spring-biz-context.xml 中的类路径中(即它在 WEB-INF/classes/com/me/config 中结束)
  2. 将 Spring Quartz 配置放在 spring-quartz-context.xml 中,在类路径中(即它也会在 WEB-INF/classes/com/me/config 中结束)
  3. 在WEB-INF中放置一个applicationContext.xml文件;让它简单地导入以上两个 XML 文件
  4. web.xml中声明一个ContextLoaderListener,这样webapp就会寻找上面提到的applicationContext.xml文件,从而初始化根上下文。
  5. web.xml 中声明一个名为 UsMain 的 servlet(DispatcherServlet 类型)
  6. 在 WEB-INF 中创建一个 UsMain-servlet.xml 文件,其中包含一些仅用于 Web 应用程序上下文的最小内容。

以下是我的配置文件,去掉了所有的 XML 文件:

UsMain-servlet.xml

    <bean id="viewResolver"
    class="org.springframework.web.servlet.view.InternalResourceViewResolver">
    <property name="prefix" value="/WEB-INF/views/" />
    <property name="suffix" value=".jsp" />
</bean>

<mvc:resources mapping="/rsc/**" location="/rsc/" cache-period="31556926"/>

<mvc:annotation-driven />

<context:component-scan base-package="com.me.controllers" />

web.xml

<web-app ... version="2.5">
...

  <servlet>
<servlet-name>UsMain</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
  </servlet>

  <servlet-mapping>
<servlet-name>UsMain</servlet-name>
<url-pattern>/</url-pattern>
  </servlet-mapping>

<listener>  
    <listener-class>  
        org.springframework.web.context.ContextLoaderListener  
    </listener-class>  
</listener>

</web-app>

applicationContext.xml

    <import resource="classes/com/me/config/spring-biz-context.xml" />
<import resource="classes/com/me/config/spring-quartz-context.xml" />

以下是我在 Tomcat 启动时看到两次的日志行示例:

2013-11-07 05:18:27 [main] INFO  org.springframework.web.context.ContextLoader - Root WebApplicationContext: initialization started
2013-11-07 05:18:27 [main] INFO  org.springframework.web.context.support.XmlWebApplicationContext - Refreshing Root WebApplicationContext: startup date [Thu Nov 07 05:18:27 UTC 2013]; root of context hierarchy
2013-11-07 05:18:27 [main] INFO  org.springframework.beans.factory.xml.XmlBeanDefinitionReader - Loading XML bean definitions from ServletContext resource [/WEB-INF/applicationContext.xml]
2013-11-07 05:18:35 [main] INFO  org.springframework.beans.factory.xml.XmlBeanDefinitionReader - Loading XML bean definitions from ServletContext resource [/WEB-INF/classes/com/me/config/spring-biz-context.xml]
... bunch of stuff ...
2013-11-07 05:18:38 [main] INFO  org.springframework.web.context.support.XmlWebApplicationContext - Refreshing WebApplicationContext for namespace 'UsMain-servlet': startup date [Thu Nov 07 05:18:38 UTC 2013]; parent: Root WebApplicationContext
2013-11-07 05:18:38 [main] INFO  org.springframework.beans.factory.xml.XmlBeanDefinitionReader - Loading XML bean definitions from ServletContext resource [/WEB-INF/UsMain-servlet.xml]

另一件事可能是也可能不是红鲱鱼......如果我以任何方式修改我的代码或配置文件,然后重新打包 WAR 文件,然后停止 Tomcat,将 WAR 复制到 webapps/ 中,然后启动 Tomcat,应用程序上下文只加载一次(耶!)但是如果我在那之后重新启动 Tomcat,就会出现这个问题。

编辑

当我不断探索这个问题时,“可能是也可能不是红鲱鱼”的事情似乎是一个重要因素。事实上,这可能更多是关于 Tomcat 的问题,而不是关于 Spring 的问题。

如果我在重新启动 Tomcat 之前从 webapps/ 目录中删除了已部署的 webapp 的子目录,那么问题不会发生。如果我在重新启动 Tomcat 之前保持子目录不变,那么问题确实发生了。

换句话说,我将 webapp 打包到 MyApp.war 中,然后将该 WAR 粘贴到 Tomcat 的 webapps/ 目录中。当 Tomcat 启动时,它会解压缩 WAR 并创建一个 webapps/MyApp/ 子目录。第一次,一切正常。但是如果我只是重新启动Tomcat,就会出现问题。如果我删除 webapps/MyApp/ 并然后重新启动 Tomcat,那么一切都很好。

所以,在某种程度上,问题解决了。但是关于正在发生的事情的想法仍然会非常有帮助。据我所知,这不是 Tomcat/webapps 的预期操作方式。

【问题讨论】:

    标签: spring tomcat spring-mvc applicationcontext


    【解决方案1】:

    我一直在研究这个问题,现在很明显问题不在于 Spring 及其应用程序上下文,而在于 Tomcat 正在做的事情。因此,我将回答我提出的问题“因为 Tomcat 在重新启动时会部署两次 Web 应用程序”。

    我的证据是,如果我在重新启动之前从 webapps/ 目录中删除我的 web 应用程序的部署子目录,则问题永远不会发生,但如果我 删除它,它总是会发生。另外,我在我的一个“Singleton”Spring bean 中添加了一些日志记录;当出现这个问题并且 bean 被创建了两次,每次都由不同的类加载器加载时。

    至于 Tomcat 为何这样做,我还不知道,但这是一个完全不同的问题。

    编辑

    只是补充一点,我已经弄清楚了 Tomcat 的问题,以防万一其他人遇到与我相同的问题。这与我尝试将我的 WAR 文件部署为根 Web 应用程序有关。查看此 URL:Apache Tomcat Configuration Reference 并搜索此字符串:

    If you want to deploy a WAR file or a directory using a context path that is not related to the base file name then one of the following options must be used to prevent double-deployment:
    

    【讨论】:

      【解决方案2】:

      尝试将 web.xml contextConfigLocation 中的 applicationContext.xml 导入直接声明为 applicationContext 的一部分。

      <context-param>
          <param-name>contextConfigLocation</param-name>
          <param-value>
              /WEB-INF/applicationContext.xml,
              /path/to/your/file1.xml,
              /path/to/your/file2.xml
          </param-value>
      </context-param>
      

      【讨论】:

      • 我很欣赏这个建议,但结果是一样的。应用程序上下文仍然如上所述加载两次。
      • mmmm,你能从 web.xml 和 applicationContext.xml 文件中编写完整的代码吗?
      • 原来是 Tomcat 配置问题(如下所述)而不是 Spring 问题。
      猜你喜欢
      • 1970-01-01
      • 2015-03-06
      • 2012-12-21
      • 2012-10-20
      • 2014-08-18
      • 2010-09-23
      • 1970-01-01
      • 2012-10-04
      • 2013-04-03
      相关资源
      最近更新 更多