学习教程框架SpringMVC入门-01
sinarcsinxSpringMVC入门-01
SpringMVC概述
Spring 为展现层提供的基于 MVC 设计理念的优秀的 Web 框架,是目前最主流的MVC 框架之一
一种轻量级的、基于MVC的Web层应用框架,它能让我们对请求数据的出来,响应数据的处理,页面的跳转等等常见的web操作变得更加简单方便
入门案例
导入相关依赖
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 
 |  <dependencies>
 <dependency>
 <groupId>javax.servlet</groupId>
 <artifactId>javax.servlet-api</artifactId>
 <version>3.1.0</version>
 <scope>provided</scope>
 </dependency>
 
 
 <dependency>
 <groupId>javax.servlet.jsp</groupId>
 <artifactId>jsp-api</artifactId>
 <version>2.1</version>
 <scope>provided</scope>
 </dependency>
 
 
 <dependency>
 <groupId>org.springframework</groupId>
 <artifactId>spring-webmvc</artifactId>
 <version>5.1.9.RELEASE</version>
 </dependency>
 
 
 <dependency>
 <groupId>com.fasterxml.jackson.core</groupId>
 <artifactId>jackson-databind</artifactId>
 <version>2.9.0</version>
 </dependency>
 </dependencies>
 
 <build>
 <plugins>
 <plugin>
 <groupId>org.apache.tomcat.maven</groupId>
 <artifactId>tomcat7-maven-plugin</artifactId>
 <version>2.2</version>
 <configuration>
 
 <port>81</port>
 
 <path>/</path>
 
 <uriEncoding>utf-8</uriEncoding>
 </configuration>
 </plugin>
 </plugins>
 </build>
 
 | 
配置web.xml
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 
 | <servlet><servlet-name>DispatcherServlet</servlet-name>
 <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
 
 
 
 
 
 
 <init-param>
 <param-name>contextConfigLocation</param-name>
 <param-value>classpath:spring-mvc.xml</param-value>
 </init-param>
 
 
 
 <load-on-startup>1</load-on-startup>
 </servlet>
 
 <servlet-mapping>
 <servlet-name>DispatcherServlet</servlet-name>
 
 
 
 
 
 <url-pattern>/</url-pattern>
 </servlet-mapping>
 
 
 
 
 <filter>
 <filter-name>CharacterEncodingFilter</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>
 </filter>
 <filter-mapping>
 <filter-name>CharacterEncodingFilter</filter-name>
 
 <url-pattern>/*</url-pattern>
 </filter-mapping>
 
 | 
配置SpringMVC
在resources目录下创建mvc的配置文件spring-mvc.xml
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 
 | 
 
 <context:component-scan base-package="com.sinarcsinx.controller"/>
 
 <mvc:default-servlet-handler/>
 
 <mvc:annotation-driven>
 <mvc:message-converters>
 <bean class="org.springframework.http.converter.StringHttpMessageConverter">
 <constructor-arg value="utf-8"/>
 </bean>
 </mvc:message-converters>
 </mvc:annotation-driven>
 
 | 
创建测试用的jsp页面
在webapp下创建success.jsp
| 12
 3
 4
 5
 6
 7
 8
 9
 
 | <%@ page contentType="text/html;charset=UTF-8" language="java" %><html>
 <head>
 <title>Title</title>
 </head>
 <body>
 成功
 </body>
 </html>
 
 | 
编写Controller
定义一个类,在类上加上@Controller注解,声明其是一个Controller。主要要创建在之前注解扫描所配置的包下
然后定义一个方法,在方法上加上@RequestMapping来指定哪些请求会被该方法所处理
| 12
 3
 4
 5
 6
 7
 8
 
 | @Controllerpublic class TestController {
 @RequestMapping("/hello")
 public String hello(){
 System.out.println("hello");
 return "/success.jsp";
 }
 }
 
 | 
设置请求映射规则@RequestMapping
该注解可以加到方法上或者是类上
我们可以用其来设定所能匹配请求的要求,只有符合了设置的要求,请求才能被加了该注解的方法或类处理
指定请求路径
path或者value属性都可以用来指定请求路径。
例如:我们期望让请求的资源路径为**/test/testPath的请求能够被testPath**方法处理则可以写如下代码
| 12
 3
 4
 5
 6
 7
 8
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 @RequestMapping("/testPath")
 public String testPath(){
 return "/success.jsp";
 }
 }
 
 | 
指定请求方式
method属性可以用来指定可处理的请求方式。
例如:我们期望让请求的资源路径为**/test/testMethod的POST请求能够被testMethod**方法处理。则可以写如下代码
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 
 @RequestMapping(value = "/testMethod",method = RequestMethod.POST)
 public String testMethod(){
 System.out.println("testMethod处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
注意:我们可以也可以运用如下注解来进行替换
-     @PostMapping    等价于   @RequestMapping(method = RequestMethod.POST)  
- 	@GetMapping    等价于   @RequestMapping(method = RequestMethod.GET)  
- 	@PutMapping    等价于   @RequestMapping(method = RequestMethod.PUT)  
- 	@DeleteMapping    等价于   @RequestMapping(method = RequestMethod.DELETE) 
例如:上面的需求我们可以使用下面的写法实现
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 
 @PostMapping(value = "/testMethod")
 public String testMethod(){
 System.out.println("testMethod处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
指定请求参数
我们可以使用params属性来对请求参数进行一些限制。可以要求必须具有某些参数,或者是某些参数必须是某个值,或者是某些参数必须不是某个值
例如:我们期望让请求的资源路径为**/test/testParams的GET请求,并且请求参数中具有code参数**的请求能够被testParams方法处理,则可以写如下代码
| 12
 3
 4
 5
 6
 7
 8
 9
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 @RequestMapping(value = "/testParams",method = RequestMethod.GET,params = "code")
 public String testParams(){
 System.out.println("testParams处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
如果是要求不能有code这个参数可以把改成如下形式
| 12
 3
 4
 5
 6
 7
 8
 9
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 @RequestMapping(value = "/testParams",method = RequestMethod.GET,params = "!code")
 public String testParams(){
 System.out.println("testParams处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
如果要求有code这参数,并且这参数值必须是某个值可以改成如下形式
| 12
 3
 4
 5
 6
 7
 8
 9
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 @RequestMapping(value = "/testParams",method = RequestMethod.GET,params = "code=sina")
 public String testParams(){
 System.out.println("testParams处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
如果要求有code这参数,并且这参数值必须不是某个值可以改成如下形式	
| 12
 3
 4
 5
 6
 7
 8
 9
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 @RequestMapping(value = "/testParams",method = RequestMethod.GET,params = "code!=sina")
 public String testParams(){
 System.out.println("testParams处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
指定请求头
我们可以使用headers属性来对请求头进行一些限制
例如:我们期望让请求的资源路径为**/test/testHeaders的GET请求,并且请求头中具有deviceType**的请求能够被testHeaders方法处理,则可以写如下代码
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 
 @RequestMapping(value = "/testHeaders",method = RequestMethod.GET,headers = "deviceType")
 public String testHeaders(){
 System.out.println("testHeaders处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
如果是要求不能有deviceType这个请求头可以把改成如下形式
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 
 @RequestMapping(value = "/testHeaders",method = RequestMethod.GET,headers = "!deviceType")
 public String testHeaders(){
 System.out.println("testHeaders处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
如果要求有deviceType这个请求头,并且其值必须是某个值可以改成如下形式
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 
 @RequestMapping(value = "/testHeaders",method = RequestMethod.GET,headers = "deviceType=ios")
 public String testHeaders(){
 System.out.println("testHeaders处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
如果要求有deviceType这个请求头,并且其值必须不是某个值可以改成如下形式
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 
 | @Controller@RequestMapping("/test")
 public class TestController {
 
 @RequestMapping(value = "/testHeaders",method = RequestMethod.GET,headers = "deviceType!=ios")
 public String testHeaders(){
 System.out.println("testHeaders处理了请求");
 return "/success.jsp";
 }
 }
 
 | 
指定请求头Content-Type
我们可以使用consumes属性来对Content-Type这个请求头进行一些限制
范例一
我们期望让请求的资源路径为**/test/testConsumes**的POST请求,并且请求头中的Content-Type头必须为 multipart/from-data 的请求能够被testConsumes方法处理,则可以写如下代码
| 12
 3
 4
 5
 
 | @RequestMapping(value = "/testConsumes",method = RequestMethod.POST,consumes = "multipart/from-data")public String testConsumes(){
 System.out.println("testConsumes处理了请求");
 return "/success.jsp";
 }
 
 | 
范例二
如果我们要求请求头Content-Type的值必须不能为某个multipart/from-data则可以改成如下形式
| 12
 3
 4
 5
 
 | @RequestMapping(value = "/testConsumes",method = RequestMethod.POST,consumes = "!multipart/from-data")public String testConsumes(){
 System.out.println("testConsumes处理了请求");
 return "/success.jsp";
 }
 
 | 
RestFul风格
RestFul是一种网络应用程序的设计风格和开发方式 
主要规则如下:
- 每一个URI代表1种资源      
- 客户端使用GET、POST、PUT、DELETE 4个表示操作方式的动词对服务端资源进行操作:GET用来获取资源,POST用来新建资源,PUT用来更新资源,DELETE用来删除资源; 
 
- 简单参数例如id等写到url路径上  例如: /user/1 HTTP GET:获取id=1的user信息      /user/1 HTTP DELETE :删除id=1的user信息     
- 复杂的参数转换成json或者xml(现在基本都是json)写到请求体中。 
获取请求参数
获取路径参数
RestFul风格的接口一些参数是在请求路径上的。类似: /user/1  这里的1就是id
如果我们想获取这种格式的数据可以使用**@PathVariable**来实现
范例一
要求定义个RestFul风格的接口,该接口可以用来根据id查询用户。请求路径要求为  /user  ,请求方式要求为GET
而请求参数id要写在请求路径上,例如  /user/1   这里的1就是id
我们可以定义如下方法,通过如下方式来获取路径参数:
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 
 | @Controllerpublic class UserController {
 
 @RequestMapping(value = "/user/{id}",method = RequestMethod.GET)
 public String findUserById( @PathVariable("id")Integer id){
 System.out.println("findUserById");
 System.out.println(id);
 return "/success.jsp";
 }
 }
 
 | 
范例二
如果这个接口,想根据id和username查询用户。请求路径要求为  /user  ,请求方式要求为GET
而请求参数id和name要写在请求路径上,例如  /user/1/zs   这里的1就是id,zs是name
我们可以定义如下方法,通过如下方式来获取路径参数:
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 
 | @Controllerpublic class UserController {
 @RequestMapping(value = "/user/{id}/{name}",method = RequestMethod.GET)
 public String findUser(@PathVariable("id") Integer id,@PathVariable("name") String name){
 System.out.println("findUser");
 System.out.println(id);
 System.out.println(name);
 return "/success.jsp";
 }
 }
 
 | 
获取请求体中的Json格式参数
RestFul风格的接口一些比较复杂的参数会转换成Json通过请求体传递过来,这种时候我们可以使用**@RequestBody**注解获取请求体中的数据
配置
SpringMVC可以帮我们把json数据转换成我们需要的类型,但是需要进行一些基本配置
SpringMVC默认会使用jackson来进行json的解析,所以我们需要导入jackson的依赖(前面我们已经导入过)
| 12
 3
 4
 5
 6
 
 | <dependency>
 <groupId>com.fasterxml.jackson.core</groupId>
 <artifactId>jackson-databind</artifactId>
 <version>2.9.0</version>
 </dependency>
 
 | 
然后还要配置注解驱动(前面已经配置过)
| 12
 
 | <mvc:annotation-driven></mvc:annotation-driven>
 
 | 
使用
范例一
要求定义个RestFul风格的接口,该接口可以用来新建用户。请求路径要求为  /user  ,请求方式要求为POST
用户数据会转换成json通过请求体传递
请求体数据
1.获取参数封装成实体对象
如果我们想把Json数据获取出来封装User对象,我们可以这样定义方法:
| 12
 3
 4
 5
 6
 7
 8
 9
 
 | @Controllerpublic class UserController {
 @RequestMapping(value = "/user",method = RequestMethod.POST)
 public String insertUser(@RequestBody User user){
 System.out.println("insertUser");
 System.out.println(user);
 return "/success.jsp";
 }
 }
 
 | 
User实体类如下:
| 12
 3
 4
 5
 6
 7
 8
 
 | @Data@NoArgsConstructor
 @AllArgsConstructor
 public class User {
 private Integer id;
 private String name;
 private Integer age;
 }
 
 | 
2.获取参数封装成Map集合
也可以把该数据获取出来封装成Map集合:
| 12
 3
 4
 5
 6
 
 | @RequestMapping(value = "/user",method = RequestMethod.POST)public String insertUser(@RequestBody Map map){
 System.out.println("insertUser");
 System.out.println(map);
 return "/success.jsp";
 }
 
 | 
范例二
如果请求体传递过来的数据是一个User集合转换成的json,Json数据可以这样定义:
| 1
 | [{"name":"sin1","age":14},{"name":"sin2","age":15},{"name":"sin3","age":16}]
 | 
方法定义:
| 12
 3
 4
 5
 6
 
 | @RequestMapping(value = "/users",method = RequestMethod.POST)public String insertUsers(@RequestBody List<User> users){
 System.out.println("insertUsers");
 System.out.println(users);
 return "/success.jsp";
 }
 
 | 
注意事项
如果需要使用**@RequestBody**来获取请求体中Json并且进行转换,要求请求头 Content-Type 的值要为: application/json 
获取QueryString格式参数
如果接口的参数是使用QueryString的格式的话,我们也可以使用SpringMVC快速获取参数
我们可以使用**@RequestParam**来获取QueryString格式的参数
使用
范例一
要求定义个接口,该接口请求路径要求为  /testRequestParam,请求方式无要求。参数为id和name和likes。使用QueryString的格式传递
1.参数单独的获取
如果我们想把id,name,likes单独获取出来可以使用如下写法:
在方法中定义方法参数,方法参数名要和请求参数名一致,这种情况下我们可以省略**@RequestParam**注解
| 12
 3
 4
 5
 6
 7
 8
 
 | @RequestMapping("/testRquestParam")public String testRquestParam(Integer id, String name, String[] likes){
 System.out.println("testRquestParam");
 System.out.println(id);
 System.out.println(name);
 System.out.println(Arrays.toString(likes));
 return "/success.jsp";
 }
 
 | 
如果方法参数名和请求参数名不一致,我们可以加上**@RequestParam**注解例如:
| 12
 3
 4
 5
 6
 7
 8
 
 | @RequestMapping("/testRquestParam")public String testRquestParam(@RequestParam("id") Integer uid,@RequestParam("name") String name, @RequestParam("likes")String[] likes){
 System.out.println("testRquestParam");
 System.out.println(uid);
 System.out.println(name);
 System.out.println(Arrays.toString(likes));
 return "/success.jsp";
 }
 
 | 
2.获取参数封装成实体对象
如果我们想把这些参数封装到一个User对象中可以使用如下写法:
| 12
 3
 4
 5
 6
 
 | @RequestMapping("/testRquestParam")public String testRquestParam(User user){
 System.out.println("testRquestParam");
 System.out.println(user);
 return "/success.jsp";
 }
 
 | 
User类定义如下:
| 12
 3
 4
 5
 6
 7
 8
 9
 
 | @Data@NoArgsConstructor
 @AllArgsConstructor
 public class User {
 private Integer id;
 private String name;
 private Integer age;
 private String[] likes;
 }
 
 | 
测试时请求url如下:
注意:实体类中的成员变量要和请求参数名对应上。并且要提供对应的set/get方法
相关注解其他属性
required
代表是否必须,默认值为true也就是必须要有对应的参数。如果没有就会报错
如果对应的参数可传可不传则可以把去设置为fasle
例如:
| 12
 3
 4
 5
 6
 7
 8
 
 | @RequestMapping("/testRquestParam")public String testRquestParam(@RequestParam(value = "id",required = false) Integer uid,@RequestParam("name") String name, @RequestParam("likes")String[] likes){
 System.out.println("testRquestParam");
 System.out.println(uid);
 System.out.println(name);
 System.out.println(Arrays.toString(likes));
 return "/success.jsp";
 }
 
 | 
defaultValue
如果对应的参数没有,我们可以用defaultValue属性设置默认值
例如:
| 12
 3
 4
 5
 6
 7
 8
 
 | @RequestMapping("/testRquestParam")public String testRquestParam(@RequestParam(value = "id",required = false,defaultValue = "777") Integer uid,@RequestParam("name") String name, @RequestParam("likes")String[] likes){
 System.out.println("testRquestParam");
 System.out.println(uid);
 System.out.println(name);
 System.out.println(Arrays.toString(likes));
 return "/success.jsp";
 }
 
 |