上图是Spring MVC的工作原理图,我们先介绍一下里面出现的核心组件
前端控制器(DispatcherServlet)它的任务是将请求发送给Spring MVC控制器,在典型的应用程序中可能会有多个控制器,因此需要根据HanderMapping查询映射关系。
处理器映射(HanderMapping)主要存储着将一个URL映射到后端控制器(Controller)的关系。
后端控制器(Controller)主要用于具体业务处理,处理完之后一般返回一个ModelAndView,包含了响应所需数据和URL信息等。
视图解析器(ViewResolver)主要用于配置视图解析引擎,如:使用freemarker作为视图模板。根据配置的信息可以找到具体的视图(View)
视图(View)用于展示用户的具体使用界面
接下来,我们对上图的具体过程做详细介绍:
①在请求离开浏览器时,会带有用户所请求内容的信息,至少包含请求的URL。但是还可能带有其他的信息,如用户提交的表单信息等。
②请求先来到了前端控制器(DispatcherServlet),前端控制器根据处理器映射(HanderMapping)查询该请求应该由哪个后端控制器(Controller)处理
③选择了合适的控制器之后,前端控制器(DispatcherServlet)会将请求发送给选中的后端控制器(Controller)。在后端控制器里面请求会卸下其负载(用户提交的信息)并耐心等待控制器处理这些信息(业务逻辑可能会被委托给一个或多个服务对象)。
④后端控制器(Controller)在完成逻辑处理之后,通常要产生一些信息,这些信息需要返回给用户并在浏览器上显示。因此后端控制器(Controller)将模型数据打包,并且标示出用于渲染输出的视图名称(一般返回一个ModelAndView),然后发送给前端控制器(DispatcherServlet)。
⑤前端控制器(DispatcherServlet)使用视图解析器(ViewResolver)来将逻辑视图名匹配为一个特定的视图实现。
⑥最后一步为视图的实现,视图使用模型数据渲染输出,并通过这个输出将响应对象传递给客户端。
开发步骤:
1、首先导入spring的jar包
2、配置web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5"
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<servlet>
<servlet-name>mvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- <init-param>如果放在WEB-INF文件下则不用此配置
<param-name>contextConfigLocation</param-name>
<param-value>classpath:mvc-servlet.xml</param-value>
</init-param> -->
</servlet>
<servlet-mapping>
<servlet-name>mvc</servlet-name>
<url-pattern>*.do</url-pattern>
</servlet-mapping>
<display-name></display-name>
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
</web-app>
3、配置mvc-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: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-4.0.xsd">
<!-- 自动扫描 -->
<context:component-scan base-package="com.hw.controller"></context:component-scan>
<!-- 试图解析器 -->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<!-- 前缀 -->
<property name="prefix" value="/WEB-INF/view/"></property>
<!-- 后缀 -->
<property name="suffix" value=".jsp"></property>
</bean>
</beans>
需要注意三点:
1、 <init-param>中指定了该servlet启动时的参数contextConfigLocation,这个参数的值用来指定Spring MVC的配置文件的位置。
2、base-package扫描的是控制层的包com.hw.controller不是类
3、<url-pattern>
写成/和/*的区别注意到,<servlet-mapping>中的<url-pattern>参数为/,而如果写成/*
又有什么区别呢?先看结论:<url-pattern>/</url-pattern>:会匹配到/springmvc
这样的**路径型**url,不会匹配到模式为*.jsp这样的**后缀型**url。
<url-pattern>/*</url-pattern>
:会匹配**所有**的url:路径型的和后缀型的url(包括/springmvc,*.jsp,*.js和*.html等)。
使用注解标识Controller类
package com.hw.controller;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import com.hw.entity.User;
@Controller //controller层中将 controller 认为是 MVC中的C --控制层
@RequestMapping("china") // 注解为控制器指定可以处理哪些 URL 请求
public class Hello {
@RequestMapping("he")//即访问url为 ../china/he.do
public String he(){
return "hello";//要返回的字符即是view文件下对应的页面
}
@RequestMapping(value="toadd")//即访问url为 ../china/toadd.do
public String toadd(){
return "add";
}
@RequestMapping("add3")//即访问url为 ../china/list.do
public String addUser3(HttpServletRequest request){
//String user,String pwd,只要方法中的参考数名和表单中元素属性名一致则自动取值
String u = request.getParameter("user");
String p = request.getParameter("pwd");
request.setAttribute("u", u);
request.setAttribute("p", p);
return "list";
}
@RequestMapping("add2")//即访问url为 ../china/list.do
public String addUser2(HttpServletRequest request,User uu){
//String user,String pwd,只要方法中的参考数名和表单中元素属性名一致则自动取值
// request.setAttribute("uu", uu);
HttpSession session = request.getSession();
session.setAttribute("uu", uu);
return "redirect:list.do";
}
@RequestMapping(value="add1")//即访问url为 ../china/list.do
public String addUser1(String user,String pwd){
//String user,String pwd,只要方法中的参考数名和表单中元素属性名一致则自动取值
System.out.println(user+ " "+pwd);
return "list";
}
@RequestMapping(value="list")//即访问url为 ../china/list.do
public String show(){
System.out.println("重定向来了 !!!");
return "list";
}
}
@Controller表示该类作为项目的控制器组件;
@RequestMapping
RequestMapping是一个用来处理请求地址映射的注解,可用于类或方法上。用于类上,表示类中的所有响应请求的方法都是以该地址作为父路径。用于方法上,表示具体路径
请求的处理过程
当有请求http://localhost:8080/springmvc/hello时,请求会被Spring MVC的DispatcherServlet拦截,并映射到HelloWorldController的sayHello()方法上处理。
返回的hello将被交给配置在spring-mvc-config.xml文件中的InternalResourceViewResolver视图处理器处理。
这个处理器的配置将controller中返回的字符串"success",加上前缀"/WEB-INF/views/jsp/",加上后缀".jsp",拼成资源的完整路径"/WEB-INF/views/jsp/helloworld.jsp",最后返回该jsp视图。