понедельник, 1 августа 2011 г.

Spring MVC - Context double initialization

Столкнулся с проблемой. Есть приложение на Spring Framework 3 (MVC + JDBC + Spring Security).
Конфигурация - частично в ХМЛ, частично - аннотациями. Однако при запуске приложения я обнаружил, что многие Бины были инициализированы дважды.
Вот мой веб - хмл:



<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.4"
         xmlns="http://java.sun.com/xml/ns/j2ee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
  http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">

  <display-name>TeklaNIS riketest teavitamine : Processor</display-name>

  <!--
  - Location of the XML file that defines the root application context
  - Applied by ContextLoaderListener.
  -->
  <context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>
      classpath:ApplicationContext.xml
      classpath:spring-security.xml
    </param-value>
  </context-param>

  <filter>
    <filter-name>springSecurityFilterChain</filter-name>
    <filter-class>org.springframework.web.filter.DelegatingFilterProxy</filter-class>
  </filter>

  <filter-mapping>
    <filter-name>springSecurityFilterChain</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>

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


  <servlet>
    <servlet-name>manager</servlet-name>
    <jsp-file>/WEB-INF/jsp/manager.jsp</jsp-file>
  </servlet>
  <servlet>
    <servlet-name>monitoring</servlet-name>
    <jsp-file>/WEB-INF/jsp/monitoring.jsp</jsp-file>
  </servlet>

  <!-- Spring integration; Configuration is inside WEB-INF/spring-servlet.xml -->
  <servlet>
    <servlet-name>spring</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <init-param>
      <param-name>contextConfigLocation</param-name>
      <param-value>
        /WEB-INF/spring-servlet.xml
      </param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
  </servlet>


  <servlet-mapping>
    <servlet-name>manager</servlet-name>
    <url-pattern>/manager</url-pattern>
  </servlet-mapping>
  <servlet-mapping>
    <servlet-name>monitoring</servlet-name>
    <url-pattern>/monitoring</url-pattern>
  </servlet-mapping>

  <servlet-mapping>
    <servlet-name>spring</servlet-name>
    <url-pattern>*.action</url-pattern>
  </servlet-mapping>

</web-app>


В принципе, вполне обычный хмл, такой, как и советуют в стандартной документации. Однако бины, которые по умолчению должны быть синглетонами, существуют в 2-х экземплярах.

Сначала я подозревал, что  DispatcherServlet и ContextLoaderListener ничего друг о друге не знают, и инициализируют 2 независимых контекста. Однако, вскоре обнаружилось, что проблема в аннотациях.
Дело в том, что и  ApplicationContext.xml, и spring-servlet.xml включали такой код:
<?xml version="1.0" encoding="UTF-8"?>
<context:annotation-config/>
<context:component-scan base-package="ee.example"/>

То есть бины, обозначенные аннотациями, обрабатывались Спрингом дважды.
Стоило мне изменить параметр base-package так, имена пакетов не перекрывались - и всё встало на свои места.

Комментариев нет:

Отправить комментарий