我有一个带有applicationContext.xml和dispatcher-servlet.xml配置的Spring Web应用程序。我已经<context:component-scan />在applicationContext.xml中定义了,但是当我运行我的应用程序时,除非同时添加<context:component-scan />到dispatcher-servlet.xml中,否则找不到控制器。我在两个中都使用了相同的基本软件包,所以这不是问题。
<context:component-scan />
我很困惑,因为我认为 applicationContext.xml是dispatcher-servlet.xml的父级。放入<context:component-scan />applicationContext.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>
编辑:我也在调度程序servlet.xml中使用mvc:annotation驱动,它应该用来拾取Controllers(我认为呢?)。
编辑2:这是配置文件。我从applicationContext.xml中删除了一堆Spring Security和OAuth设置(出于安全原因,并且它们可能根本不相关)。
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>
dispatcher-servlet.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>
编辑3:好的,这很有趣。我的服务和dao类位于从Web项目引用的其他项目(JAR)中。我正在使用基于Java的配置,并从applicationContext.xml引用它:
<bean class="bar.foo.config.ServicesConfig" />
因此,这意味着我的Web项目(位于applicationContext.xml所在的位置)中仅存在Controller批注。回顾一下,从我的applicationContext.xml中删除context:component-scan应该不会有任何影响,因为除了@Controller以外,没有其他注释(FIX到EDIT:有一些@Autowired注释)。但是,当我从applicationContext.xml中删除context:component-scan时,它说控制器(从调度程序servlet扫描中找到)找不到我的Service类。对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时,控制器没有选择自动装配的服务bean。这是因为根上下文引用了我的基于Java服务的配置Bean,但是我没有根上下文设置来扫描组件。因此,当我将组件扫描添加到根上下文(applicationContext.xml)时,一切正常。这是我现在所拥有的:
applicationContext.xml:
<bean class="bar.foo.config.ServicesConfig" /> <context:component-scan base-package="bar.foo.config" />
dispatcher-servlet.xml:
<context:component-scan base-package="bar.foo.controller" />
我已在Web上下文中设置了Controller,Autowired以及Controller程序包中的所有其他注释-我不确定这是否是最佳实践。
没错-有两个不同的应用程序上下文,由ContextLoaderListener加载的根应用程序上下文(在ServletContext初始化时)和Web上下文(由DispatcherServlet加载),根应用程序上下文是Web的父级上下文。
现在,由于这是两个不同的应用程序上下文,因此它们的作用方式有所不同-如果你component-scan在应用程序上下文中定义服务,则将在此处创建服务的所有bean。
component-scan
当你的Dispatcher Servlet加载时,它将开始创建Web上下文,在某个时候(由<mvc:annotation-driven/>它驱动将为你的uri到处理程序方法创建映射,它将获取应用程序上下文中的bean列表(将是Web应用程序)上下文,而不是Root应用程序上下文),并且由于尚未在component-scan此处定义,因此将找不到与控制器相关的Bean,也不会创建映射,这就是为什么必须在调度程序Servlet上下文中定义组件扫描的原因也。
<mvc:annotation-driven/>
一个好的做法是在“根应用程序上下文”中排除与Controller相关的bean:
<context:component-scan base-package="package"> <context:exclude-filter expression="org.springframework.stereotype.Controller" type="annotation"/> </context:component-scan>
在Web应用程序上下文中,只有与控制器相关的一个:
<context:component-scan base-package="package" use-default-filters="false"> <context:include-filter expression="org.springframework.stereotype.Controller" type="annotation" /> </context:component-scan>