applicationContext не находит контроллеры для контекста сервлета

у меня есть весеннее веб-приложение с applicationContext.xml и диспетчер-сервлет.конфигурация xml. Я определил <context:component-scan /> в applicationContext.xml, но когда я запускаю свое приложение, контроллеры не найдены, если я также не добавлю <context:component-scan /> диспетчер-сервлета.XML. Я использую один и тот же базовый пакет в обоих, так что это не проблема.

Я смущен, потому что я мысль что applicationContext.xml был родителем диспетчера-сервлета.XML. Не положить <context:component-scan /> в applicationContext.XML-файле достаточно?

web.в XML

<?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">


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

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

<servlet>
    <servlet-name>dispatcher</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <init-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/dispatcher-servlet.xml</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
</servlet>

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

</web-app>

EDIT:я также использую MVC: annotation-driven в dispatcher-servlet.xml, который должен забрать контроллеры (я думал?).

EDIT 2: Вот файлы конфигурации. Я удалил кучу настроек Spring Security и OAuth из applicationContext.xml (по соображениям безопасности и, вероятно, они не актуальны в любом случае.)

applicationContext.в XML

<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:sec="http://www.springframework.org/schema/security" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p" xmlns:c="http://www.springframework.org/schema/c"
xmlns:context="http://www.springframework.org/schema/context" xmlns:oauth="http://www.springframework.org/schema/security/oauth2"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
      http://www.springframework.org/schema/security http://www.springframework.org/schema/security/spring-security-3.1.xsd
      http://www.springframework.org/schema/security/oauth2 http://www.springframework.org/schema/security/spring-security-oauth2.xsd
      http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd">

<context:component-scan base-package="bar.foo"/>
<context:property-placeholder location="classpath:my.properties" />
<bean class="bar.foo.ServicesConfig" />

</beans>

диспетчер-сервлета.в XML

<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p" xmlns:c="http://www.springframework.org/schema/c"
xmlns:context="http://www.springframework.org/schema/context" xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
      http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd
      http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd">

<context:component-scan base-package="bar.foo.controller" />
<mvc:annotation-driven/>
<mvc:default-servlet-handler />

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

<bean id="contentViewResolver" class="org.springframework.web.servlet.view.ContentNegotiatingViewResolver">
    <property name="mediaTypes">
        <map>
            <entry key="json" value="application/json" />
        </map>
    </property>
    <property name="defaultViews">
        <bean class="org.springframework.web.servlet.view.json.MappingJacksonJsonView" />
    </property>
    <property name="order" value="1" />
</bean>

</beans>

EDIT 3: Хорошо, это интересно. Мои службы и классы dao находятся в другом проекте (JAR), на который я ссылаюсь из веб-проекта. Я использую конфигурацию на основе java и ссылаюсь на нее из applicationContext.XML-код:

<bean class="bar.foo.config.ServicesConfig" />

Итак, это означает, что в моем веб-проекте есть только аннотации контроллера (где applicationContext.xml находится). В ретроспектива, удаление контекста: компонент-сканирование из моего applicationContext.xml не должен влиять, так как нет аннотаций, кроме @Controller (исправление для редактирования: есть некоторые аннотации @Autowired). Но, когда я удаляю контекст: component-scan из applicationContext.xml, он говорит, что контроллеры (найденные из сканирования сервлетов диспетчера) не могут найти мои классы обслуживания. Разве ссылки на ServicesConfig недостаточно? Вот класс ServicesConfig для ссылок - он имеет собственное сканирование компонентов для служб, которые отличаются от пакета applicationContext.XML был сканирования.

@Configuration
@ComponentScan({ "some.other.package", "another.package" })
@ImportResource({ "classpath:commonBeans.xml" })
@PropertySource({ "classpath:services.properties",
"classpath:misc.properties" })
public class ServicesConfig {
  // Bean definitions //
}

устранение:

когда я удалил context: component-scan из моего корневого контекста, контроллеры не собирали компоненты autowired services. Это было связано с тем, что корневой контекст ссылается на мой компонент конфигурации на основе java, но у меня не было настройки корневого контекста для сканирования компонентов. Следовательно, когда я добавляю сканирование компонентов к корневому контексту (applicationContext.xml) все работает. Вот что у меня теперь:

applicationContext.XML-код:

<bean class="bar.foo.config.ServicesConfig" />
<context:component-scan base-package="bar.foo.config" />

диспетчер-сервлета.XML-код:

<context:component-scan base-package="bar.foo.controller" />

у меня есть настройка веб - контекста для контроллера пикапа, Autowired и любых других аннотаций в пакете контроллера-я не уверен, что это лучшая практика или нет.

3 ответов


вы правы - есть два разных контекста приложения, корневой контекст приложения, загруженный ContextLoaderListener (в момент инициализации ServletContext), и веб-контекст (загруженный DispatcherServlet), корневой контекст приложения является родителем веб-контекста.

теперь, поскольку это два разных контекста приложения, они действуют по-разному - если вы определяете component-scan для ваших услуг в контексте приложения, затем все компоненты для услуг, которые создаются здесь.

когда ваш сервлет диспетчера загрузится, он начнет создавать веб-контекст, в какой-то момент(управляемый <mvc:annotation-driven/> он создаст сопоставление для методов обработчика вашего uri, он получит список бобов в контексте приложения (который будет контекстом веб-приложения, а не корневым контекстом приложения), и поскольку вы не определили component-scan здесь связанные с контроллером бобы не будут найдены, и сопоставления не будут созданы, то есть причина, по которой вы должны определить компонент-сканирование в контексте сервлетов диспетчера также.

хорошей практикой является исключение связанных с контроллером компонентов в контексте корневого приложения:

<context:component-scan base-package="package">
    <context:exclude-filter expression="org.springframework.stereotype.Controller" type="annotation"/>
</context:component-scan>

и только контроллер, связанный с ним в контексте веб-приложения:

<context:component-scan base-package="package" use-default-filters="false">
    <context:include-filter expression="org.springframework.stereotype.Controller" type="annotation" />
</context:component-scan>

в наших приложениях мы определяем в dispatcher-servlet.в XML

Я считаю, что именно там он должен быть, а не в applicationContext.в XML

этот раздел документации весны должен обеспечить больше информации:

http://static.springsource.org/spring/docs/current/spring-framework-reference/html/mvc.html

Как вы можете видеть из одной из диаграмм в разделе 16.2, диспетчер-сервлет сидит выше applicationContext в иерархии контекста.


у меня была та же проблема и после сравнения моего web.xml код в этом уроке Я изменил его и он работал. вот мой :

<web-app xmlns="http://java.sun.com/xml/ns/javaee"
     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
     xmlns:mvc="http://www.springframework.org/schema/mvc"
     xsi:schemaLocation="
    http://java.sun.com/xml/ns/javaee
    http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
     version="3.0">
<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath:spring/business-config.xml</param-value>
</context-param>

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

<servlet>
    <servlet-name>mvc-dispatcher</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
</servlet>

<servlet-mapping>
    <servlet-name>mvc-dispatcher</servlet-name>
    <url-pattern>/</url-pattern>
</servlet-mapping>

context-param и listener было то, что я пропустил.

Я надеюсь, это поможет вам.