Spring MVC:<context:component-scan> 和 <annotation-driven /> 标签之间的区别? [复制]

Posted

技术标签:

【中文标题】Spring MVC:<context:component-scan> 和 <annotation-driven /> 标签之间的区别? [复制]【英文标题】:Spring MVC: difference between <context:component-scan> and <annotation-driven /> tags? [duplicate] 【发布时间】:2012-11-19 15:55:31 【问题描述】:

前几天开始研究这个Spring Hello World教程:http://viralpatel.net/blogs/spring-3-mvc-create-hello-world-application-spring-3-mvc/

在本教程中,Spring DispatcherServlet 是使用 spring-servlet.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:p="http://www.springframework.org/schema/p"
    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 
    http://www.springframework.org/schema/context/spring-context.xsd">

<context:component-scan base-package="net.viralpatel.spring3.controller" />

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

在这个文件中,我使用 context:component-scan 标签来表示 Spring 必须扫描我的文件以搜索注解,例如,当控制器类发现某个方法被注解时通过 @RequestMapping("/hello") 注释知道该方法处理向以“/hello”结尾的 URL 的 HTTP 请求。这很简单……

现在我的疑问与我可以在 STS\Eclipse 中自动构建的 Spring MVC 模板项目有关。

当我在 STS 中创建一个新的 Spring MVC 项目时,我的 DispatcherServlet 是由一个名为 servlet-context.xml 的文件配置的,该文件包含一些类似于上一个示例文件。

在这个文件中,我还有组件扫描标签:

<context:component-scan base-package="com.mycompany.maventestwebapp" />

但我还有另一个标签(看起来有类似的任务),这个:

<annotation-driven />

这两个标签有什么区别? 另一个“奇怪”的事情是前面的示例(不使用注释驱动标签)与 STS 使用 Spring MVC 模板项目创建的项目非常相似,但是如果我从其配置中删除注释驱动标签文件项目不运行并给我以下错误:HTTP Status 404 -

在堆栈跟踪中我有:

警告:org.springframework.web.servlet.PageNotFound - 在名为“appServlet”的 DispatcherServlet 中找不到具有 URI [/maventestwebapp/] 的 HTTP 请求的映射

但是为什么呢?前面的例子在没有注释驱动标签的情况下运行良好,这个控制器类非常相似。实际上,只有一种方法可以处理对“/”路径的 HTTP 请求

这是我的控制器类的代码:

package com.mycompany.maventestwebapp;

import java.text.DateFormat;
import java.util.Date;
import java.util.Locale;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

/**
 * Handles requests for the application home page.
*/
@Controller
public class HomeController 

private static final Logger logger = LoggerFactory.getLogger(HomeController.class);

/**
 * Simply selects the home view to render by returning its name.
 */
@RequestMapping(value = "/", method = RequestMethod.GET)
public String home(Locale locale, Model model) 
    logger.info("Welcome home! The client locale is .", locale);

    Date date = new Date();
    DateFormat dateFormat = DateFormat.getDateTimeInstance(DateFormat.LONG, DateFormat.LONG, locale);

    String formattedDate = dateFormat.format(date);

    model.addAttribute("serverTime", formattedDate );

    return "home";

有人可以帮我理解这件事吗?

非常感谢!

【问题讨论】:

非常类似于***.com/questions/3977973/…。这个答案很好***.com/questions/3977973/… 有趣,但不完全符合我的疑惑 【参考方案1】:

&lt;mvc:annotation-driven /&gt; 表示您可以定义spring beans 依赖项,而无需实际在XML 中指定一堆元素或实现接口或扩展基类。例如@Repository 告诉 spring 一个类是一个 Dao,而不必扩展 JpaDaoSupport 或 DaoSupport 的一些其他子类。类似地,@Controller 告诉 spring,指定的类包含将处理 Http 请求的方法,而无需实现 Controller 接口或扩展实现控制器的子类。

当 spring 启动时,它读取它的 XML 配置文件并在其中查找 &lt;bean 元素,如果它看到类似 &lt;bean class="com.example.Foo" /&gt; 并且 Foo 被标记为 @Controller 它知道该类是一个控制器并处理它因此。默认情况下,Spring 假定它应该管理的所有类都明确定义在 beans.XML 文件中。

使用&lt;context:component-scan base-package="com.mycompany.maventestwebapp" /&gt; 进行组件扫描告诉spring 它应该在类路径中搜索com.mycompany.maventestweapp 下的所有类,并查看每个类是否有@Controller@Repository 或@ 987654331@ 或 @Component 如果是这样,那么 Spring 将向 bean 工厂注册该类,就像您在 XML 配置文件中键入 &lt;bean class="..." /&gt; 一样。

在一个典型的 spring MVC 应用程序中你会发现有两个 spring 配置文件,一个配置应用程序上下文的文件通常以 Spring 上下文侦听器开始。

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

Spring MVC 配置文件通常以 Spring 调度程序 servlet 启动。例如。

<servlet>
        <servlet-name>main</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>main</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

Spring 支持分层 bean 工厂,因此在 Spring MVC 的情况下,调度程序 servlet 上下文是主应用程序上下文的子级。如果向 servlet 上下文询问名为“abc”的 bean,它将首先在 servlet 上下文中查找,如果在其中找不到它,它将在父上下文中查找,即应用程序上下文。

数据源、JPA 配置、业务服务等公共 bean 是在应用程序上下文中定义的,而 MVC 特定配置不是与 servlet 关联的配置文件。

希望这会有所帮助。

【讨论】:

好吧,这很清楚,但我不明白为什么其他网站示例不使用注释驱动标签,而 STS 示例需要它... Spring 并不总是有注释和组件扫描,因此您可能会在网上看到各种示例,展示各种不同的技术和组合。 还需要 标签吗?我刚刚测试了一个没有 的 Spring 3.0 演示应用程序,只有 标记,我的应用程序工作正常!!!!事实上——这些天几乎没有教程把 放在那里 spring-servlet.xml【参考方案2】:

注释驱动向 Spring 指示它应该扫描带注释的 bean,而不是仅仅依赖 XML bean 配置。组件扫描指示在哪里查找这些 bean。

这是一些文档:http://static.springsource.org/spring/docs/current/spring-framework-reference/html/mvc.html#mvc-config-enable

【讨论】:

【参考方案3】:
<context:component-scan base-package="" /> 

告诉 Spring 扫描这些包的注解。

<mvc:annotation-driven> 

注册一个RequestMappingHanderMapping、一个RequestMappingHandlerAdapter和一个ExceptionHandlerExceptionResolver来支持MVC自带的@RequestMapping、@ExceptionHandler等带注解的控制器方法。

这也启用了一个 ConversionService,它支持注解驱动的输出格式以及注解驱动的输入验证。它还支持@ResponseBody,您可以使用它返回 JSON 数据。

您可以使用基于 Java 的配置在 @Configuration 类中使用 @ComponentScan(basePackages="...", "..." 和 @EnableWebMvc 来完成相同的事情。

查看 3.1 文档以了解更多信息。

http://static.springsource.org/spring/docs/3.1.x/spring-framework-reference/html/mvc.html#mvc-config

【讨论】:

以上是关于Spring MVC:<context:component-scan> 和 <annotation-driven /> 标签之间的区别? [复制]的主要内容,如果未能解决你的问题,请参考以下文章

请问,java高手,spring mvc拦截器如何拦截所有的请求啊,包括html和jsp页面?

spring mvc json乱码

Spring MVC Interceptor

Spring 一二事 - IOC MVC 简易搭建

Spring Mvc拦截器不起作用

Spring MVC