Весна MVC: разница между и теги? [дубликат]
этот вопрос уже есть ответ здесь:
- Почему Spring MVC отвечает 404 и сообщает "нет сопоставления для HTTP-запроса с URI [ ... ] в DispatcherServlet"? 3 ответы
несколько дней назад я начал изучать этот весенний учебник Hello World: http://viralpatel.net/blogs/spring-3-mvc-create-hello-world-application-spring-3-mvc/
в этом уроке Spring DispatcherServlet настраивается с помощью spring-сервлет.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>
в этом файле я использую контекст: component-scan тег, чтобы сказать, что Spring должен сканировать мой файл, ищущий аннотацию, так, например, когда класс контроллера обнаруживает, что метод аннотация @RequestMapping ("/hello") аннотация знает, что этот метод обрабатывает HTTP-запрос к URL-адресу, заканчивающемуся на "/hello". Это очень просто...
теперь мое сомнение связано с проектом шаблона Spring MVC, который я мог бы автоматически построить в STSEclipse.
когда я создаю новый проект Spring MVC в STS у меня есть, что мой DispatcherServlet настраивается с помощью файла с именем сервлет-контексте.xml, который содержит некоторые конфигурация аналогична предыдущему файлу примера.
в этом файле у меня все еще есть тег сканирования компонентов:
<context:component-scan base-package="com.mycompany.maventestwebapp" />
но у меня есть еще один тег (который выглядит как аналогичная задача), этот:
<annotation-driven />
в чем разница между этими двумя тегами?
Другая "странная" вещь заключается в том, что предыдущий пример (который не использует тег, управляемый аннотациями) очень похож на проект, созданный STS с использованием проекта шаблона Spring MVC, но если я удалите тег, управляемый аннотациями, из файла конфигурации, который проект не запускает, и дайте мне следующую ошибку:HTTP-статус 404 -
и в stacktrace у меня есть:
WARN: org.springframework.сеть.сервлет.PageNotFound-не найдено сопоставления для HTTP-запроса с URI [/maventestwebapp/] в DispatcherServlet с именем 'appServlet'
но почему? Предыдущий пример хорошо работает без тега, управляемого аннотациями, и это класс контроллера очень похож. На самом деле, есть только один метод, который обрабатывает HTTP-запрос к "/" path
это код моего класса контроллера:
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";
}
может кто-нибудь помочь мне понять эту вещь?
большое спасибо!
3 ответа:
<mvc:annotation-driven />
означает, что вы можете определить зависимости spring beans без фактического указания группы элементов в XML или реализации интерфейса или расширения базового класса. Например@Repository
чтобы сказать spring, что класс является Dao без необходимости расширенияJpaDaoSupport
или какой-то другой подкласс DaoSupport. Точно так же@Controller
сообщает spring, что указанный класс содержит методы, которые будут обрабатывать Http-запросы без необходимости реализации интерфейса контроллера или расширения подкласса это реализует контроллер.при запуске spring он читает свой XML-файл конфигурации и ищет
<bean
элементы внутри него, если он видит что-то вроде<bean class="com.example.Foo" />
и Foo был выделен@Controller
Он знает, что класс является контроллером и рассматривает его как таковой. По умолчанию Spring предполагает, что все классы, которыми он должен управлять, явно определены в компонентах.XML-файл.сканирование компонентов с помощью
<context:component-scan base-package="com.mycompany.maventestwebapp" />
говорит весна, что она должна искать путь к классам для всех классов в com.название_компании.maventestweapp и посмотрите на каждый класс, чтобы увидеть, если он имеет@Controller
или@Repository
или@Service
или@Component
и если это произойдет, то Spring зарегистрирует класс с фабрикой бобов, как если бы вы набрали<bean class="..." />
в файлах конфигурации XML.в типичном приложении spring MVC вы обнаружите, что есть два файла конфигурации spring, файл, который настраивает контекст приложения, обычно запускаемый с помощью прослушивателя контекста Spring.
<listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener>
и файл конфигурации Spring MVC обычно запускается с сервлета Spring dispatcher. Например.
<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 поддерживает иерархические фабрики бобов, поэтому в случае Spring MVC контекст сервлета диспетчера является дочерним для основного контекста приложения. Если контекст сервлета был запрошен для компонента с именем "abc", он сначала будет выглядеть в контексте сервлета, если он не найдет его там, он будет выглядеть в Родительском контексте, который является контекст приложения.
общие компоненты, такие как источники данных, конфигурация JPA, бизнес-службы, определяются в контексте приложения, в то время как конкретная конфигурация MVC не является файлом конфигурации, связанным с сервлетом.
надеюсь, что это помогает.
<context:component-scan base-package="" />
говорит Spring сканировать эти пакеты для аннотаций.
<mvc:annotation-driven>
регистрирует RequestMappingHanderMapping, RequestMappingHandlerAdapter и ExceptionHandlerExceptionResolver для поддержки аннотированных методов контроллера, таких как @RequestMapping, @ExceptionHandler и т. д. это приходит с MVC.
Это также позволяет ConversionService, который поддерживает аннотацию управляемое форматирование выходных данных, а также аннотацию управляемой проверки для входных данных. Это также включает поддержку @ResponseBody, которую можно использовать для возврата данных JSON.
вы можете выполнить то же самое, используя конфигурацию на основе Java, используя @ComponentScan(basePackages={"...","..."} и @EnableWebMvc в классе @Configuration.
ознакомьтесь с документацией 3.1, чтобы узнать больше.
http://static.springsource.org/spring/docs/3.1.x/spring-framework-reference/html/mvc.html#mvc-config
Annotation-driven указывает Spring, что он должен сканировать аннотированные бобы, а не просто полагаться на конфигурацию XML-бобов. Компонент-сканирование указывает, где искать эти бобы.
вот некоторые док: http://static.springsource.org/spring/docs/current/spring-framework-reference/html/mvc.html#mvc-config-enable