1. ajax异步交互
Springmvc默认用MappingJackson2HttpMessageConverter对json数据进行转换,需要加入jackson的包;同时使用 <mvc:annotation-driven />
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-databind</artifactId> <version>2.9.8</version> </dependency> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-core</artifactId> <version>2.9.8</version> </dependency> <dependency> <groupId>com.fasterxml.jackson.core</groupId> <artifactId>jackson-annotations</artifactId> <version>2.9.0</version> </dependency>
|
1.1 @RequestBody
该注解用于Controller的方法的形参声明,当使用ajax提交并指定contentType为json形式时,通过HttpMessageConverter接口转换为对应的POJO对象。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| <button id="btn1"> ajax异步提交 </button> <script> $("#btn1").click(function() { let url = '${pageContext.request.contextPath}/ajaxRequest'; let data = '[{"id":1,"username":"张三"},{"id":2,"username":"李四"}]'; $.ajax({ type: 'POST', url: url, data: data, contentType: 'application/json;charset=utf-8', success: function(resp) { alert(JSON.stringify(resp)) } }) }) </script>
|
1 2 3 4
| @RequestMapping(value = "/ajaxRequest") public void ajaxRequest(@RequestBody List<User>list) { System.out.println(list); }
|
1.2 @ResponseBody
该注解用于将Controller的方法返回的对象,通过HttpMessageConverter接口转换为指定格式的数据如:json,xml等,通过Response响应给客户端。
1 2 3 4 5 6 7 8 9 10
|
@RequestMapping(value = "/ajaxRequest") @ResponseBody public List<User> ajaxRequest(@RequestBody List<User> list) { System.out.println(list); return list; }
|
2. RESTful
2.1 什么是RESTful
Restful是一种软件架构风格、设计风格,而不是标准,只是提供了一组设计原则和约束条件。主要用于客户端和服务器交互类的软件,基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存机制等。
Restful风格的请求是使用“url+请求方式”表示一次请求目的的,HTTP 协议里面四个表示操作方式的动词如下:
- GET:读取(Read)
- POST:新建(Create)
- PUT:更新(Update)
- DELETE:删除(Delete)
客户端请求 |
原来风格URL地址 |
RESTful风格URL地址 |
查询所有 |
/user/findAll |
GET /user |
根据ID查询 |
/user/findById?id=1 |
GET /user/{1} |
新增 |
/user/save |
POST /user |
修改 |
/user/update |
PUT /user |
删除 |
/user/delete?id=1 |
DELETE /user/{1} |
2.2 代码实现
@PathVariable
用来接收RESTful风格请求地址中占位符的值
@RestController
RESTful风格多用于前后端分离项目开发,前端通过ajax与服务器进行异步交互,我们处理器通常返回的是json数据所以使用@RestController来替代@Controller和@ResponseBody两个注解。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| @RestController public class RestFulController { @GetMapping(value = "/user/{id}") public String get(@PathVariable Integer id) { return "get:" + id; } @PostMapping(value = "/user") public String post() { return "post"; } @PutMapping(value = "/user") public String put() { return "put"; } @DeleteMapping(value = "/user/{id}") public String delete(@PathVariable Integer id) { return "delete:"+ id; } }
|
3. 文件上传
3.1 文件上传三要素
- 表单项 type=”file”
- 表单的提交方式 method=”POST”
- 表单的enctype属性是多部分表单形式 enctype=“multipart/form-data”

3.2 文件上传原理
- 当form表单修改为多部分表单时,request.getParameter()将失效。
- 当form表单的enctype取值为 application/x-www-form-urlencoded 时,form表单的正文内容格式是: name=value&name=value
- 当form表单的enctype取值为 mutilpart/form-data 时,请求正文内容就变成多部分形式:

3.3 单文件上传
步骤分析
- 导入fileupload和io坐标
- 配置文件上传解析器
- 编写文件上传代码
1)导入fileupload和io坐标
1 2 3 4 5 6 7 8 9 10
| <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>1.3.3</version> </dependency> <dependency> <groupId>commons-io</groupId> <artifactId>commons-io</artifactId> <version>2.6</version> </dependency>
|
2)配置文件上传解析器
1 2 3 4 5 6 7
| <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> <property name="maxUploadSize" value="5242880"></property> <property name="maxInMemorySize" value="40960"></property> </bean>
|
3)编写文件上传代码
1 2 3 4 5
| <form action="${pageContext.request.contextPath}/fileUpload" method="post" enctype="multipart/form-data"> 名称:<input type="text" name="username"><br> 文件:<input type="file" name="filePic"><br> <input type="submit" value="单文件上传"> </form>
|
1 2 3 4 5 6 7 8 9
| @RequestMapping("/fileUpload") public String fileUpload(String username, MultipartFile filePic) throws IOException { System.out.println(username); String originalFilename = filePic.getOriginalFilename(); filePic.transferTo(new File("d:/upload/"+originalFilename)); return "success"; }
|
3.4 多文件上传
1 2 3 4 5 6
| <form action="${pageContext.request.contextPath}/filesUpload" method="post" enctype="multipart/form-data"> 名称:<input type="text" name="username"><br> 文件1:<input type="file" name="filePic"><br> 文件2:<input type="file" name="filePic"><br> <input type="submit" value="多文件上传"> </form>
|
1 2 3 4 5 6 7 8 9 10 11
| @RequestMapping("/filesUpload") public String filesUpload(String username, MultipartFile[] filePic) throws IOException { System.out.println(username); for (MultipartFile multipartFile : filePic) { String originalFilename = multipartFile.getOriginalFilename(); multipartFile.transferTo(new File("d:/upload/" + originalFilename)); } return "success"; }
|
4. 异常处理
4.1 异常处理的思路
在Java中,对于异常的处理一般有两种方式:
- 一种是当前方法捕获处理(try-catch),这种处理方式会造成业务代码和异常处理代码的耦合。
- 另一种是自己不处理,而是抛给调用者处理(throws),调用者再抛给它的调用者,也就是一直向上抛。在这种方法的基础上,衍生出了SpringMVC的异常处理机制。
系统的dao、service、controller出现都通过throws Exception向上抛出,最后由springmvc前端控制器交由异常处理器进行异常处理,如下图:

4.2 自定义异常处理器
步骤分析
- 创建异常处理器类实现HandlerExceptionResolver
- 配置异常处理器
- 编写异常页面
- 测试异常跳转
1)创建异常处理器类实现HandlerExceptionResolver
1 2 3 4 5 6 7 8 9
| public class GlobalExceptionResolver implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { ModelAndView modelAndView = new ModelAndView(); modelAndView.addObject("error", ex.getMessage()); modelAndView.setViewName("error"); return modelAndView; } }
|
2)配置异常处理器
1 2
| @Component public class GlobalExecptionResovler implements HandlerExceptionResolver {}
|
3)编写异常页面
1 2 3 4 5 6 7 8 9 10
| <%@ page contentType="text/html;charset=UTF-8" language="java" %> <html> <head> <title>error</title> </head> <body> <h3>这是一个最终异常的显示页面</h3> <p>${error}</p> </body> </html>
|
4)测试异常跳转
1 2 3 4 5
| @RequestMapping("/testException") public String testException() { int i = 1 / 0; return "success"; }
|
4.3 web的处理异常机制
1 2 3 4 5 6 7 8 9 10
| <error-page> <error-code>500</error-code> <location>/500.jsp</location> </error-page>
<error-page> <error-code>404</error-code> <location>/404.jsp</location> </error-page>
|
5. 拦截器
5.1 拦截器(interceptor)的作用
Spring MVC 的拦截器类似于 Servlet 开发中的过滤器 Filter,用于对处理器进行预处理和后处理。
将拦截器按一定的顺序联结成一条链,这条链称为拦截器链(InterceptorChain)。在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序被调用。拦截器也是AOP思想的具体实现。
5.2 拦截器和过滤器区别
关于interceptor和filter的区别,如图所示:

5.3 快速入门
步骤分析
- 创建拦截器类实现HandlerInterceptor接口
- 配置拦截器
- 测试拦截器的拦截效果
1)创建拦截器类实现HandlerInterceptor接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| public class MyInterceptor1 implements HandlerInterceptor { @Override public Boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) { System.out.println("preHendle1"); return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) { System.out.println("postHandle1"); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { System.out.println("afterCompletion1"); } }
|
2)配置拦截器
1 2 3 4 5 6 7 8
| <mvc:interceptors> <mvc:interceptor> <mvc:mapping path="/**"/> <bean class="com.lagou.interceptor.MyInterceptor1"/> </mvc:interceptor> </mvc:interceptors>
|
3)测试拦截器的拦截效果
编写Controller,发请求到controller,跳转页面
1 2 3 4 5 6 7 8
| @Controller public class TargetController { @RequestMapping("/target") public String targetMethod() { System.out.println("目标方法执行了..."); return "success"; } }
|
编写jsp页面
1 2 3 4 5 6 7 8 9 10
| <%@ page contentType="text/html;charset=UTF-8" language="java" %> <html> <head> <title>success</title> </head> <body> <h3>success...</h3> <% System.out.println( "视图执行了....");%> </body> </html>
|
5.4 拦截器链
开发中拦截器可以单独使用,也可以同时使用多个拦截器形成一条拦截器链。开发步骤和单个拦截器是一样的,只不过注册的时候注册多个,注意这里注册的顺序就代表拦截器执行的顺序。
同上,再编写一个MyHandlerInterceptor2操作,测试执行顺序:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| <mvc:interceptors> <mvc:interceptor> <mvc:mapping path="/**"/> <bean class="com.lagou.interceptor.MyInterceptor1"></bean> </mvc:interceptor> <mvc:interceptor> <mvc:mapping path="/**"/> <bean class="com.lagou.interceptor.MyInterceptor2"></bean> </mvc:interceptor> </mvc:interceptors>
|
5.5 知识小结
拦截器中的方法说明如下:
