一、SpringMVC 基础
1、三层架构
表现层:
- 也就是我们常说的web层。
- 它负责接收客户端请求,向客户端响应结果,通常客户端使用http协议请求web 层,web 需要接收 http 请求,完成 http 响应。
- 表现层包括展示层和控制层:控制层负责接收请求,展示层负责结果的展示。表现层依赖业务层,接收到客户端请求一般会调用业务层进行业务处理,并将处理结果响应给客户端。
- 表现层的设计一般都使用 MVC 模型。(MVC 是表现层的设计模型,和其他层没有关系)
业务层:
- 也就是我们常说的 service 层。
- 它负责业务逻辑处理,和我们开发项目的需求息息相关。web 层依赖业务层,但是业务层不依赖 web 层。
- 业务层在业务处理时可能会依赖持久层,如果要对数据持久化需要保证事务一致性。(也就是我们说的,事务应该放到业务层来控制)
持久层:
- 也就是我们是常说的 dao 层。
- 负责数据持久化,包括数据层即数据库和数据访问层,数据库是对数据进行持久化的载体,数据访问层是业务层和持久层交互的接口,业务层需要通过数据访问层将数据持久化到数据库中。通俗的讲,持久层就是和数据库交互,对数据库表进行曾删改查的。
2、MVC 模型
MVC 模型是一种用于设计创建 Web 应用程序表现层的模式。
Model(模型):
- 通常指的就是我们的数据模型。作用一般情况下用于封装数据。
View(视图):
- 通常指的就是我们的 jsp 或者 html。作用一般就是展示数据的。
- 通常视图是依据模型数据创建的。
Controller(控制器):
- 是应用程序中处理用户交互的部分。作用一般就是处理程序逻辑的。
Controller(控制器)相对于前两个不是很好理解,这里举个例子:
我们要保存一个用户的信息,该用户信息中包含了姓名,性别,年龄等等。这时候表单输入要求年龄必须是 1~100 之间的整数。姓名和性别不能为空。并且把数据填充到模型之中。此时除了 js 的校验之外,服务器端也应该有数据准确性的校验,那么校验就是控制器的该做的。 当校验失败后,由控制器负责把错误页面展示给使用者。如果校验成功,也是控制器负责把数据填充到模型,并且调用业务层实现完整的业务需求。
3、SpringMVC 概述
- SpringMVC 是一种基于 Java 的实现 MVC 设计模型的请求驱动类型的轻量级 Web 框架。
- Spring是框架,MVC 是一种设计模式。
- Spring 框架提供了构建 Web 应用程序的全功能 MVC 模块。
- 使用 Spring 可插入的 MVC 架构,从而在使用 Spring 进行 WEB 开发时,可以选择使用 Spring 的 Spring MVC 框架或集成其他 MVC 开发框架。
4、SpringMVC 在三层架构的位置
5、SpringMVC 的优势
- 清晰的角色划分:
前端控制器(DispatcherServlet
)
请求到处理器映射(HandlerMapping
)
处理器适配器(HandlerAdapter
)
视图解析器(ViewResolver
)
处理器或页面控制器(Controller
)
验证器(Validator
)
命令对象(Command
请求参数绑定到的对象就叫命令对象)
表单对象(Form Object
提供给表单展示和提交到的对象就叫表单对象)。 - 分工明确,而且扩展点相当灵活,可以很容易扩展,虽然几乎不需要。
- 由于命令对象就是一个 POJO,无需继承框架特定 API,可以使用命令对象直接作为业务对象。
- 和 Spring 其他框架无缝集成,是其它 Web 框架所不具备的。
- 可适配,通过 HandlerAdapter 可以支持任意的类作为处理器。
- 可定制性,HandlerMapping、ViewResolver 等能够非常简单的定制。
- 功能强大的数据验证、格式化、绑定机制。
- 利用 Spring 提供的 Mock 对象能够非常简单的进行 Web 层单元测试。
- 本地化、主题的解析的支持,使我们更容易进行国际化和主题的切换。
- 强大的 JSP 标签库,使 JSP 编写更容易。
二、示例代码
第一步:配置核心控制器,一个 Servlet
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0">
<!--配置SpringMVC核心控制器-->
<servlet>
<servlet-name>SpringMVC</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!--配置初始化参数,用于读取SpringMVC的配置文件-->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:/SpringMVC.xml</param-value><!--文件位置在resource中-->
</init-param>
<!--配置 servlet 的对象的创建时间点:应用加载时创建,取值只能是非 0 的正整数,表示启动顺序-->
<load-on-startup>1</load-on-startup>
</servlet>
<!--映射路径-->
<servlet-mapping>
<servlet-name>SpringMVC</servlet-name><!--取值与上面的值对应-->
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
第二步:创建 spring mvc 的配置文件,SpringMVC.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"
xmlns:mvc="http://www.springframework.org/schema/mvc"
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 http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd">
<!--配置 Spring 容器要要扫描的包-->
<context:component-scan base-package="cn.lemon"/>
<!--配置处理器映射器,作用是根据路径查找处理器-->
<!--<bean id="handlerMapping" class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"/>-->
<!--处理器适配器,作用是执行具体的处理器-->
<!--<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"/>-->
<!--
一般情况下,处理器映射器、处理器适配器可以省略,也可以用 <mvc:annotation-driven> 来代替
mvc:annotation-driven:作用就是代替处理器映射器和处理器适配器
-->
<mvc:annotation-driven></mvc:annotation-driven>
<!--配置视图解析器-->
<bean id="resolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<!--
通过前缀和后缀来解析这个视图
prefix:前缀
suffix:后缀
-->
<property name="prefix" value="/WEB-INF/pages/"/>
<property name="suffix" value=".jsp"/>
</bean>
</beans>
第三步:编写控制器并使用注解配置
package cn.lemon;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
/**
* HelloController 相当于一个 servlet,
* 相比于 servlet,HelloController 可以放入 Spring 容器中,可以使用依赖注入与控制反转
* servlet 不可以放入 Spring 容器中,因为 servlet 是tomcat 产生的
*/
@Controller
public class HelloController {
@RequestMapping("/hello")
public String sayHello() {
System.out.println("hello SpringMVC...........");
return "success";//转发到 WEB-INF/pages/(前缀)success(名字).jsp(后缀)(这个文件路径是在 SpringMVC.xml 视图解析器中加上前缀和后缀生成的)
}
}
第四步:新建展示页面 index.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>主页</title>
</head>
<body>
<a href="${pageContext.request.contextPath}/hello">点击跳转</a>
</body>
</html>
第五步:在WEB-INF/pages 中新建跳转页面 success.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>成功</title>
</head>
<body>
<h3>成功!!!!</h3>
</body>
</html>
三、SpringMVC 执行原理
1、案例的执行过程
1、服务器启动,应用被加载。读取到 web.xml 中的配置创建 spring 容器并且初始化容器中的对象。从上面的案例中可以看到的是:HelloController
和InternalResourceViewResolver
,但是远不止这些。
2、浏览器发送请求,被 DispatherServlet
捕获,该 Servlet
并不处理请求,而是把请求转发出去。转发的路径是根据请求 URL,匹配@RequestMapping
中的内容。
3、匹配到了后,执行对应方法。该方法有一个返回值。
4、根据方法的返回值,借助 InternalResourceViewResolver
找到对应的结果视图。
5、渲染结果视图,响应浏览器。
2、SpringMVC 的请求响应流程
3、上面案例中的组件分析
1、DispatcherServlet:前端控制器
- 用户请求到达前端控制器,它就相当于 mvc 模式中的 c,dispatcherServlet
是整个流程控制的中心,由它调用其它组件处理用户的请求,dispatcherServlet 的存在降低了组件之间的耦合性。
2、HandlerMapping:处理器映射器
- HandlerMapping 负责根据用户请求找到 Handler 即处理器,SpringMVC
提供了不同的映射器实现不同的映射方式,例如:配置文件方式,实现接口方式,注解方式等。
3、Handler:处理器
- 它就是我们开发中要编写的具体业务控制器。由 DispatcherServlet 把用户请求转发到 Handler。由Handler对具体的用户请求进行处理。
4、HandlAdapter:处理器适配器
- 通过 HandlerAdapter 对处理器进行执行,这是适配器模式的应用,通过扩展适配器可以对更多类型的处理器进行执行。
5、View Resolver:视图解析器
- View Resolver 负责将处理结果生成 View 视图,View Resolver
首先根据逻辑视图名解析成物理视图名即具体的页面地址,再生成 View 视图对象,最后对 View 进行渲染将处理结果通过页面展示给用户。
6、View:视图
- SpringMVC 框架提供了很多的 View 视图类型的支持,包括:jstlView、freemarkerView、pdfView等。我们最常用的视图就是 jsp。一般情况下需要通过页面标签或页面模版技术将模型数据通过页面展示给用户,需要由程序员根据业务需求开发具体的页面。
7、<mvc:annotation-driven>
说明
- 在 SpringMVC 的各个组件中,处理器映射器、处理器适配器、视图解析器称为 SpringMVC 的三大组件。
- 使 用
<mvc:annotation-driven>
自动加载RequestMappingHandlerMapping
(处理映射器) 和RequestMappingHandlerAdapter
( 处 理 适 配 器 ) , 可 用 在 SpringMVC.xml 配 置 文 件 中 使 用<mvc:annotation-driven>
替代注解处理器和适配器的配置
8、SpringMVC 框架基于组件方式执行流程
4、RequestMapping 注解
示例:
jsp代码
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>主页</title>
</head>
<body>
<h3>案例</h3>
<a href="${pageContext.request.contextPath}/hello">点击跳转</a><hr/>
<h3>添加</h3>
<a href="${pageContext.request.contextPath}/account/add">点击跳转</a><hr/>
<h3>Get请求,删除</h3>
<a href="${pageContext.request.contextPath}/account/delete">点击跳转</a><hr/><%--会报错,报 405--%>
<h3>Post请求,删除</h3>
<form method="post" action="${pageContext.request.contextPath}/account/delete"><%--会执行成功--%>
<input type="submit" value="提交">
</form><hr/>
<h3>Get请求,修改</h3>
<a href="${pageContext.request.contextPath}/account/update">点击跳转</a><hr/><%--会报 404--%>
<h3>Get请求,修改</h3>
<a href="${pageContext.request.contextPath}/account/update?aid=2">点击跳转</a><hr/><%--会报 404--%>
<h3>Get请求,修改</h3>
<a href="${pageContext.request.contextPath}/account/update?id=1">点击跳转</a><hr/><%--会执行成功--%>
</body>
</html>
java代码
package cn.lemon;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
@Controller
//请求 URL 的第一级访问目录。此处不写的话,就相当于应用的根目录。写的话需要以/开头。
@RequestMapping("/account")
public class AccountController {
@RequestMapping("/add")
public String add() {
System.out.println("添加账户。。。。。。。。。。。");
return "success";
}
@RequestMapping(value = "/delete", method = RequestMethod.POST)
public String delete() {
System.out.println("删除账户。。。。。。。。");
return "success";
}
/*
* 属性:
value:用于指定请求的 URL。它和 path 属性的作用是一样的。
method:用于指定请求的方式。
params:用于指定限制请求参数的条件。它支持简单的表达式。要求请求参数的 key 和 value 必须和配置的一模一样。
* */
@RequestMapping(value = "/update", method = {RequestMethod.POST, RequestMethod.GET}, params = {"id"})
public String update() {
System.out.println("修改账户。。。。。。。。");
return "success";
}
}
四、请求参数的绑定
1、绑定的机制
我们都知道,表单中请求参数都是基于 key=value 的。SpringMVC 绑定请求参数的过程是通过把表单提交请求参数,作为控制器中方法参数进行绑定的。
2、支持的数据类型
- 基本类型参数:包括基本类型和 String 类型
- POJO 类型参数:包括实体类,以及关联的实体类
- 数组和集合类型参数:包括 List 结构和 Map 结构的集合(包括数组)
3、使用要求
- 如果是基本类型或者 String 类型:要求我们的参数名称必须和控制器中方法的形参名称保持一致。(严格区分大小写)
- 如果是 POJO 类型,或者它的关联对象:要求表单中参数名称和 POJO 类的属性名称保持一致。并且控制器方法的参数类型是 POJO 类型。
- 如果是集合类型,有两种方式:
-
- 第一种:
要求集合类型的请求参数必须在 POJO 中。在表单中请求参数名称要和 POJO 中集合属性名称相同。
给 List 集合中的元素赋值,使用下标。
给 Map 集合中的元素赋值,使用键值对。
- 第一种:
-
- 第二种:
接收的请求参数是 json 格式数据。需要借助一个注解实现。
- 第二种:
4、示例代码——基本类型作为参数
jsp 代码
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>主页</title>
</head>
<body>
<h2>基本类型参数绑定</h2>
<a href="${pageContext.request.contextPath}/account/add?accountId=1&accountName=lemon">点击跳转</a><hr/>
</body>
</html>
java控制器代码
package cn.lemon;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("/account")
public class AccountController {
@RequestMapping("/add")
public String add(Integer accountId, String accountName){
System.out.println("添加账户,添加账户的 id 为:" + accountId + ",添加账户的姓名为:" + accountName);
return "success";
}
}
5、示例代码——POJO 类型作为参数
定义实体类 Account.java、User.java
package cn.lemon.domain;
import java.io.Serializable;
public class Account implements Serializable {
private String username;
private String password;
private Double money;
private User user;
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
public Double getMoney() {
return money;
}
public void setMoney(Double money) {
this.money = money;
}
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
@Override
public String toString() {
return "Account{" +
"username='" + username + '\'' +
", password='" + password + '\'' +
", money=" + money +
", User=" + user +
'}';
}
}
package cn.lemon.domain;
import java.io.Serializable;
import java.util.Date;
public class User implements Serializable {
private String uname;
private Integer age;
private Date birthday;
public String getUname() {
return uname;
}
public void setUname(String uname) {
this.uname = uname;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public Date getBirthday() {
return birthday;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
@Override
public String toString() {
return "User{" +
"uname='" + uname + '\'' +
", age=" + age +
", birthday=" + birthday +
'}';
}
}
jsp页面代码
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>主页</title>
</head>
<body>
<h2>POJO 实体类</h2>
<form action="${pageContext.request.contextPath}/account/add" method="post">
姓名:<input type="text" name="username"/><br>
密码:<input type="text" name="password"/><br>
金额:<input type="text" name="money"/><br>
用户姓名:<input type="text" name="user.uname"/><br>
用户年龄:<input type="text" name="user.age"/><br>
<input type="submit" value="提交"/>
</form>
</body>
</html>
java控制器代码
package cn.lemon.controller;
import cn.lemon.domain.Account;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
@Controller
@RequestMapping("/account")
public class AccountController {
@RequestMapping(value = "/add",method = RequestMethod.POST)
public String add(Account account){
System.out.println("添加成功" + account);
return "success";
}
}
6、示例代码——数组和集合类型作为参数
在User.java 实体类中添加集合,并生成get 、set 方法
private List<Account> accountList = new ArrayList<>();
private Map<String, Account> accountMap = new HashMap<>();
jsp页面代码
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>主页</title>
</head>
<body>
<h2>封装参数到集合(list,map)</h2>
<form action="${pageContext.request.contextPath}/user/add" method="post">
用户姓名:<input type="text" name="uname" /><br/>
用户年龄:<input type="text" name="age" /><br/>
账号1名字:<input type="text" name="accountList[0].username" /><br/>
账号1金额:<input type="text" name="accountList[0].money" /><br/>
账号2名字:<input type="text" name="accountList[1].username" /><br/>
账号2金额:<input type="text" name="accountList[1].money" /><br/>
账号3名字:<input type="text" name="accountList[2].username" /><br/>
账号3金额:<input type="text" name="accountList[2].money" /><br/>
<hr>
map1名字:<input type="text" name="accountMap['one'].username" /><br/>
map1金额:<input type="text" name="accountMap['one'].money" /><br/>
map2名字:<input type="text" name="accountMap['two'].username" /><br/>
map2金额:<input type="text" name="accountMap['two'].money" /><br/>
<input type="submit" value="提交" />
</form>
</body>
</html>
java控制器代码
package cn.lemon.controller;
import cn.lemon.domain.User;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("/user")
public class UserController {
@RequestMapping(value = "/add")
public String add(User user) {
System.out.println("添加用户..." + user);
return "success";
}
}
7、请求参数乱码问题
1、post 请求方式:
在 web.xml 中配置一个 SpringMVC 编码过滤器
<!--配置 SpringMVC 编码过滤器-->
<filter>
<filter-name>filter</filter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<!--设置过滤器中的属性值-->
<init-param>
<param-name>encoding</param-name>
<param-value>UTF-8</param-value>
</init-param>
<!--启动过滤器,默认情况下是启动的,可以省略-->
<init-param>
<param-name>forceEncoding</param-name>
<param-value>true</param-value>
</init-param>
</filter>
<!--过滤所有请求-->
<filter-mapping>
<filter-name>filter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
2、get 请求方式:
tomacat 对 GET 和 POST 请求处理方式是不同的,GET 请求的编码问题,要改 tomcat 的 server.xml 配置文件(Tomcat 8 已经解决)
8、自定义类型转换器
第一步:定义一个类 StringToDateConverter.java,实现 Converter 接口
package cn.lemon.utils;
import org.springframework.core.convert.converter.Converter;
import org.springframework.util.StringUtils;
import java.text.SimpleDateFormat;
import java.util.Date;
public class StringToDateConverter implements Converter<String, Date> {
@Override
public Date convert(String s) {
try {
if (StringUtils.isEmpty(s)) {
throw new NullPointerException("请输入日期");
}
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
return sdf.parse(s);
} catch (Exception e) {
throw new RuntimeException("输入的日期错误");
}
}
}
第二步:在 spring 配置文件中(SpringMVC.xml)配置类型转换器
spring 配置类型转换器的机制是,将自定义的转换器注册到类型转换服务中去。
<!--配置类型转换器工厂-->
<bean id="conversionServiceFactoryBean" class="org.springframework.context.support.ConversionServiceFactoryBean">
<!--给工厂注入一个新的类型转换器-->
<property name="converters">
<array>
<!--配置自定义类型转换器-->
<bean class="cn.lemon.utils.StringToDateConverter"/>
</array>
</property>
</bean>
第三步:在 annotation-driven 标签中引用配置的类型转换服务
<mvc:annotation-driven conversion-service="conversionServiceFactoryBean"></mvc:annotation-driven>
8、SpringMVC 使用 Servlet API
jsp代码
<a href="${pageContext.request.contextPath}/servletapi">点击跳转</a>
java控制器代码
@RequestMapping("/servletapi")
public String servletApi(HttpServletRequest request, HttpServletResponse response, HttpSession session, PrintWriter printWriter) throws Exception {
System.out.println(request);
System.out.println(response);
System.out.println(session);
//response.getWriter().print("hello servlet ......");
printWriter.print("hello servlet .......");
return null;
}
五、常用的注解
1、RequestParam
作用: 把请求中指定名称的参数给控制器中的形参赋值。
属性:
value:请求参数中的名称。
required:请求参数中是否必须提供此参数。默认值:true。表示必须提供,如果不提供将报错。
jsp代码
<h2>RequestParam 注解</h2>
<a href="${pageContext.request.contextPath}/user/requestparam?name=lemon&location=beijing&age=18">点击访问</a>
java 代码
package cn.lemon.controller;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
@Controller
@RequestMapping("/user")
public class UserController {
@RequestMapping("/requestparam")
//第一个@RequestParam(name = "name") 是可以省略的
// 第二个是不可以省略的,
// 第三个 @RequestParam 中 required = true,意思是必须要传入的参数,默认 required = false
//第四个 @RequestParam ,是设置默认值
public String requestParam(
@RequestParam(name = "name") String name,
@RequestParam(name = "location") String address,
@RequestParam(required = true) Integer age,
@RequestParam(defaultValue = "lemon") String usernmae){
System.out.println("RequestParam , name = " + name + ",address = " + address + ",age =" + age + ",username = " + usernmae);
return "success";
}
}
2、RequestBody
作用: 用于获取请求体内容。直接使用得到是 key=value&key=value...
结构的数据。get 请求方式不适用。
属性:
required:是否必须有请求体。默认值是:true。当取值为 true 时,get 请求方式会报错。如果取值为 false,get 请求得到是 null。
jsp 页面
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>主页</title>
</head>
<script src="${pageContext.request.contextPath}/js/jquery-2.1.0.min.js" charset="utf-8"></script>
<body>
<h2>RequestBody 注解</h2>
<form action="${pageContext.request.contextPath}/user/requestbody" method="post">
姓名:<input type="text" name="username"><br>
密码:<input type="password" name="password"><br>
年龄:<input type="text" name="age"><br>
<input type="submit" value="保存">
</form>
<h2>RequestBody2 注解</h2>
<input type="button" value="提交ajax" id="btn">
<script>
$(function () {
$("#btn").click(function () {
var json = {'username':'lemon','password':'1223','age':'18'};//定义一个json对象
var string_json = JSON.stringify(json);//把json 对象转换成一个字符串
/*使用 Ajax 发送 HTTP 请求*/
$.ajax({
method:"POST",
url:"${pageContext.request.contextPath}/user/requestbody2",
contentType:"application/json;charset=UTF-8",
data:string_json,
success:function (result) {
alert(result);
}
})
})
})
</script>
</body>
</html>
java页面
package cn.lemon.controller;
import cn.lemon.domain.User;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("/user")
public class UserController {
@RequestMapping("/requestbody")
//@RequestBody 可以把请求体转换为字符串
public String requestBody(@RequestBody String body) {
System.out.println("body = " + body);
return "success";
}
@RequestMapping("/requestbody2")
//@RequestBody 可以把请求体转换为user 对象
//不过,@RequestBody 需要借助 Jackson 框架(Maven引入 jackson 如上图),把json字符串序列化成实体对象
public String requestBody2(@RequestBody User user) {
System.out.println(user);
return null;
}
}
在SpringMVC.xml中配置静态资源不过滤
<!--
在 springmvc 的配置文件中可以配置,静态资源不过滤 (因为上面过滤掉了所有请求):
location 表示路径,mapping 表示文件,**表示该目录下的文件以及子目录的文件
-->
<mvc:resources location="/css/" mapping="/css/**"/>
<mvc:resources location="/images/" mapping="/images/**"/>
<mvc:resources location="/scripts/" mapping="/javascript/**"/>
3、PathVariable
作用: 用于绑定 url 中的占位符。例如:请求 url 中 /delete/{id}
,这个{id}
就是 url 占位符。url 支持占位符是 spring3.0 之后加入的。是 springmvc 支持 rest 风格 URL 的一个重要标志。
属性:
value:用于指定 url 中占位符名称。
required:是否必须提供占位符。
jsp 代码
<a href="${pageContext.request.contextPath}/user/pathvariable/100">PathVariable</a>
java 代码
@RequestMapping("/pathvariable/{id}")
public String pathVariable(@PathVariable("id") Integer id) {
System.out.println("添加成功,id = " + id);
return "success";
}
4、REST 风格 URL
REST(英文:Representational State Transfer,简称 REST)描述了一个架构样式的
restful 的示例:
/account
HTTP POST: 新增 account/account/1
HTTP GET : 得到 id = 1 的 account/account/1
HTTP DELETE: 删除 id = 1 的 account/account/1
HTTP PUT: 更新 id = 1 的 account
基于 HiddentHttpMethodFilter 的示例
作用:
- 由于浏览器 form 表单只支持 GET 与 POST 请求,而 DELETE、PUT 等 method 并不支持,Spring3.0添加了一个过滤器,可以将浏览器请求改为指定的请求方式,发送给我们的控制器方法,使得支持 GET、POST、PUT 与 DELETE 请求。
使用方法:
- 第一步:在 web.xml 中配置该过滤器。
- 第二步:请求方式必须使用 post 请求。
- 第三步:按照要求提供_method 请求参数,该参数的取值就是我们需要的请求方式。
web.xml 中的过滤器
<!--使浏览器支持 rest 风的 URL-->
<filter>
<filter-name>hiddenHttpMethodFilter</filter-name>
<filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
</filter>
<!--映射路径-->
<filter-mapping>
<filter-name>hiddenHttpMethodFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
jsp代码
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>主页</title>
</head>
<body>
<h2>rest 风格的 URL</h2>
<form action="${pageContext.request.contextPath}/rest/add" method="post">
<input type="submit" value="新增">
</form>
<form action="${pageContext.request.contextPath}/rest/delete/100" method="post">
<input type="hidden" name="_method" value="DELETE">
<input type="submit" value="删除">
</form>
<form action="${pageContext.request.contextPath}/rest/update/100" method="post">
<input type="hidden" name="_method" value="PUT">
<input type="submit" value="更新">
</form>
<a href="${pageContext.request.contextPath}/rest/find/100">查询</a>
</body>
</html>
java 代码
package cn.lemon.controller;
import cn.lemon.domain.Account;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
@Controller
@RequestMapping("/rest")
public class RestController {
/*
/account HTTP POST: 添加 account
/account/1 HTTP GET : 查询 id = 1 的 account
/account/1 HTTP DELETE: 删除 id = 1 的 account
/account/1 HTTP PUT: 更新 id = 1 的 account
*/
@RequestMapping(value = "/add", method = RequestMethod.POST)
public String add(Account account) {
System.out.println("添加用户,用户为:" + account);
return "success";
}
@RequestMapping(value = "/find/{id}", method = RequestMethod.GET)
public String find(@PathVariable("id") Integer id) {
System.out.println("查询用户,用户的id 为:" + id);
return "success";
}
@RequestMapping(value = "/delete/{id}", method = RequestMethod.DELETE)
public String delete(@PathVariable("id") Integer id) {
System.out.println("删除用户,id 为:" + id);
return "redirect:/index.jsp";//提醒:这里需要重定向,不能转发
}
@RequestMapping(value = "/update/{id}", method = RequestMethod.PUT)
public String update(@PathVariable("id") Integer id) {
System.out.println("更新用户,更新的id 为:" + id);
return "redirect:/index.jsp";//提醒:这里需要重定向,不能转发
}
}
5、RequestHeader
作用: 用于获取请求消息头。
属性:
value:提供消息头名称
required:是否必须有此消息头
jsp 代码
<h2>RequestHeader 获取请求头</h2>
<a href="${pageContext.request.contextPath}/other/header">获取请求头</a>
java 代码
package cn.lemon.controller;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestHeader;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("/other")
public class OtherController {
@RequestMapping("/header")
public String requestHeader(@RequestHeader(value = "Accept-Language",required = false) String requestHeader){
System.out.println("获取请求头消息,消息为:" + requestHeader);
return "success";
}
}
6、CookieValue
作用: 用于把指定 cookie 名称的值传入控制器方法参数。
属性:
value:指定 cookie 的名称。
required:是否必须有此 cookie。
jsp 代码
<h2>CookieValue 传Cookie的值</h2>
<a href="${pageContext.request.contextPath}/other/cookie">Cookie</a>
java 代码
@RequestMapping("/cookie")
public String cookieValue(@CookieValue(value = "JSESSIONID",required = false) String cookieValue) {
System.out.println("获取 Cookie 的值," + cookieValue);
return "success";
}
7、ModelAttribute
作用: 该注解是 SpringMVC4.3 版本以后新加入的。它可以用于修饰方法和参数。出现在方法上,表示当前方法会在控制器的方法执行之前,先执行。它可以修饰没有返回值的方法,也可以修饰有具体返回值的方法。出现在参数上,获取指定的数据给参数赋值。
属性:
value:用于获取数据的 key。key 可以是 POJO 的属性名称,也可以是 map 结构的 key。
应用场景:
当表单提交数据不是完整的实体类数据时,保证没有提交数据的字段使用数据库对象原来的数据。
例如:
我们在编辑一个用户时,用户有一个创建信息字段,该字段的值是不允许被修改的。在提交表单数据是肯定没有此字段的内容,一旦更新会把该字段内容置为 null,此时就可以使用此注解解决问题。
jsp 代码
<h2>ModelAttribute 在控制器方法之前执行</h2>
<a href="${pageContext.request.contextPath}/other/model">ModelAttribute</a>
java 代码
@ModelAttribute
public void showModel(User user) {
System.out.println("这是被 @ModelAttribute 修饰的方法,方法名为 showModel" + user);
}
@RequestMapping("/model")
public String modelAttribute(User user) {
System.out.println("执行了控制器的方法");//会执行被 @ModelAttribute 修饰的方法
return "success";
}
8、SessionAttribute
作用: 用于多次执行控制器方法间的参数共享。
属性:
value:用于指定存入的属性名称
type:用于指定存入的数据类型。