1 异步调用
1.1 发送异步请求(回顾)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| <a href="javascript:void(0);" id="testAjax">访问controller</a> <script type="text/javascript" src="/js/jquery-3.3.1.min.js"></script> <script type="text/javascript"> $(function(){ $("#testAjax").click(function(){ $.ajax({ type:"POST", url:"ajaxController", data:'ajax message', dataType:"text", contentType:"application/text", }); }); }); </script>
|
1.2 接受异步请求参数
名称: @RequestBody
类型: 形参注解
位置:处理器类中的方法形参前方
作用:将异步提交数据组织成标准请求参数格式,并赋值给形参
范例:
1 2 3 4 5
| @RequestMapping("/ajaxController") public String ajaxController(@RequestBody String message){ System.out.println(message); return "page.jsp"; }
|
- 注解添加到Pojo参数前方时,封装的异步提交数据按照Pojo的属性格式进行关系映射
- 注解添加到集合参数前方时,封装的异步提交数据按照集合的存储结构进行关系映射
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| @RequestMapping("/ajaxPojoToController")
public String ajaxPojoToController(@RequestBody User user){ System.out.println("controller pojo :"+user); return "page.jsp"; }
@RequestMapping("/ajaxListToController")
public String ajaxListToController(@RequestBody List<User> userList){ System.out.println("controller list :"+userList); return "page.jsp"; }
|
1.3 异步请求接受响应数据
- 方法返回值为Pojo时,自动封装数据成json对象数据
1 2 3 4 5 6 7 8 9
| @RequestMapping("/ajaxReturnJson") @ResponseBody public User ajaxReturnJson(){ System.out.println("controller return json pojo..."); User user = new User(); user.setName("Jockme"); user.setAge(40); return user; }
|
- 方法返回值为List时,自动封装数据成json对象数组数据
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| @RequestMapping("/ajaxReturnJsonList") @ResponseBody
public List ajaxReturnJsonList(){ System.out.println("controller return json list..."); User user1 = new User(); user1.setName("Tom"); user1.setAge(3);
User user2 = new User(); user2.setName("Jerry"); user2.setAge(5);
ArrayList al = new ArrayList(); al.add(user1); al.add(user2);
return al; }
|
2 异步请求-跨域访问
2.1 跨域访问介绍
- 当通过域名A下的操作访问域名B下的资源时,称为跨域访问
- 跨域访问时,会出现无法访问的现象

2.2 跨域环境搭建
- 为当前主机添加备用域名
- 修改windows安装目录中的host文件
- 格式: ip 域名
- 动态刷新DNS
- 命令: ipconfig /displaydns
- 命令: ipconfig /flushdns
2.3 跨域访问支持
名称: @CrossOrigin
类型: 方法注解 、 类注解
位置:处理器类中的方法上方 或 类上方
作用:设置当前处理器方法/处理器类中所有方法支持跨域访问
范例:
1 2 3 4 5 6 7 8 9 10 11 12 13
| @RequestMapping("/cross") @ResponseBody
@CrossOrigin public User cross(HttpServletRequest request){ System.out.println("controller cross..."+request.getRequestURL()); User user = new User(); user.setName("Jockme"); user.setAge(39); return user; }
|
3 拦截器
3.1 拦截器概念

拦截器( Interceptor)是一种动态拦截方法调用的机制
作用:
1. 在指定的方法调用前后执行预先设定后的的代码
2. 阻止原始方法的执行
核心原理: AOP思想
拦截器链:多个拦截器按照一定的顺序,对原始被调用功能进行增强
3.2 自定义拦截器开发过程
实现HandlerInterceptor接口
1 2 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
| public class MyInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { System.out.println("前置运行----a1"); return true; }
@Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { System.out.println("后置运行----b1"); }
@Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { System.out.println("完成运行----c1"); }
}
|
配置拦截器
配置拦截器
1 2 3 4 5 6
| <mvc:interceptors> <mvc:interceptor> <mvc:mapping path="/showPage"/> <bean class="com.itheima.interceptor.MyInterceptor"/> </mvc:interceptor> </mvc:interceptors>
|
注意:配置顺序为先配置执行位置,后配置执行类
3.3 拦截器执行流程

3.4 拦截器配置与方法参数
3.4.1 前置处理方法
原始方法之前运行
1 2 3 4 5 6
| public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { System.out.println("preHandle"); return true; }
|
- 参数
request:请求对象
response:响应对象
handler:被调用的处理器对象,本质上是一个方法对象,对反射中的Method对象进行了再包装
- 返回值
返回值为false,被拦截的处理器将不执行
3.4.2 后置处理方法
原始方法运行后运行,如果原始方法被拦截,则不执行
1 2 3 4 5 6
| public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { System.out.println("postHandle"); }
|
参数
modelAndView:如果处理器执行完成具有返回结果,可以读取到对应数据与页面信息,并进行调整
3.4.3 完成处理方法
拦截器最后执行的方法,无论原始方法是否执行
1 2 3 4 5 6
| public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { System.out.println("afterCompletion"); }
|
参数
ex:如果处理器执行过程中出现异常对象,可以针对异常情况进行单独处理
3.5 拦截器配置项
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| <mvc:interceptors> <mvc:interceptor> <mvc:mapping path="/*"/> <mvc:mapping path="/**"/> <mvc:mapping path="/handleRun*"/> <mvc:exclude-mapping path="/b*"/> <bean class="MyInterceptor"/> </mvc:interceptor> </mvc:interceptors>
|
3.6 多拦截器配置

责任链模式
责任链模式是一种行为模式
特征:
沿着一条预先设定的任务链顺序执行,每个节点具有独立的工作任务
优势:
独立性:只关注当前节点的任务,对其他任务直接放行到下一节点
隔离性:具备链式传递特征,无需知晓整体链路结构,只需等待请求到达后进行处理即可
灵活性:可以任意修改链路结构动态新增或删减整体链路责任
解耦:将动态任务与原始任务解耦
弊端:
链路过长时,处理效率低下
可能存在节点上的循环引用现象,造成死循环,导致系统崩溃
4 异常处理
4.1 异常处理器
HandlerExceptionResolver接口(异常处理器)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| @Component public class ExceptionResolver implements HandlerExceptionResolver { public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { System.out.println("异常处理器正在执行中"); ModelAndView modelAndView = new ModelAndView(); modelAndView.addObject("msg","出错啦! "); modelAndView.setViewName("error.jsp"); return modelAndView; } }
|
根据异常的种类不同,进行分门别类的管理,返回不同的信息
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| public class ExceptionResolver implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { System.out.println("my exception is running ...."+ex); ModelAndView modelAndView = new ModelAndView(); if( ex instanceof NullPointerException){ modelAndView.addObject("msg","空指针异常"); }else if ( ex instanceof ArithmeticException){ modelAndView.addObject("msg","算数运算异常"); }else{ modelAndView.addObject("msg","未知的异常"); } modelAndView.setViewName("error.jsp"); return modelAndView; } }
|
4.2 注解开发异常处理器
- 使用注解实现异常分类管理
名称: @ControllerAdvice
类型: 类注解
位置:异常处理器类上方
作用:设置当前类为异常处理器类
范例:
1 2 3 4
| @Component @ControllerAdvice public class ExceptionAdvice { }
|
- 使用注解实现异常分类管理
名称: @ExceptionHandler
类型: 方法注解
位置:异常处理器类中针对指定异常进行处理的方法上方
作用:设置指定异常的处理方式
范例:
说明:处理器方法可以设定多个 1 2 3 4 5
| @ExceptionHandler(Exception.class) @ResponseBody public String doOtherException(Exception ex){ return "出错啦,请联系管理员! "; }
|
4.3 异常处理解决方案
- 异常处理方案
- 业务异常:
发送对应消息传递给用户,提醒规范操作
- 系统异常:
发送固定消息传递给用户,安抚用户
发送特定消息给运维人员,提醒维护
记录日志
- 其他异常:
发送固定消息传递给用户,安抚用户
发送特定消息给编程人员,提醒维护
纳入预期范围内
记录日志
4.4 自定义异常
5 实用技术
5.1 文件上传下载
上传文件过程分析

MultipartResolver接口
MultipartResolver接口定义了文件上传过程中的相关操作,并对通用性操作进行了封装
MultipartResolver接口底层实现类CommonsMultipartResovler
CommonsMultipartResovler并未自主实现文件上传下载对应的功能,而是调用了apache的文件上传下载组件
1 2 3 4 5
| <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>1.4</version> </dependency>
|
文件上传下载实现
1 2 3 4
| <form action="/fileupload" method="post" enctype="multipart/form-data"> 上传LOGO: <input type="file" name="file"/><br/> <input type="submit" value="上传"/> </form>
|
1 2 3
| <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> </bean>
|
1 2 3 4
| @RequestMapping(value = "/fileupload") public void fileupload(MultipartFile file){ file.transferTo(new File("file.png")); }
|
5.2 文件上传注意事项
- 文件命名问题, 获取上传文件名,并解析文件名与扩展名
- 文件名过长问题
- 文件保存路径
- 重名问题
1 2 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
| @RequestMapping(value = "/fileupload")
public String fileupload(MultipartFile file,MultipartFile file1,MultipartFile file2, HttpServletRequest request) throws IOException { System.out.println("file upload is running ..."+file); if(!file.isEmpty()){ String fileName = file.getOriginalFilename(); String realPath = request.getServletContext().getRealPath("/images"); file.transferTo(new File(realPath,file.getOriginalFilename())); } if(!file1.isEmpty()){ String fileName = file1.getOriginalFilename(); String realPath = request.getServletContext().getRealPath("/images"); file1.transferTo(new File(realPath,file1.getOriginalFilename())); } if(!file2.isEmpty()){ String fileName = file2.getOriginalFilename(); String realPath = request.getServletContext().getRealPath("/images"); file2.transferTo(new File(realPath,file2.getOriginalFilename())); } return "page.jsp"; }
|
5.4 Restful风格配置
5.4.1 Rest
- Rest( REpresentational State Transfer) 一种网络资源的访问风格,定义了网络资源的访问方式
- Restful是按照Rest风格访问网络资源
- 优点
隐藏资源的访问行为,通过地址无法得知做的是何种操作
书写简化
5.4.2 Rest行为约定方式
GET(查询) http://localhost/user/1 GET
POST(保存) http://localhost/user POST
PUT(更新) http://localhost/user PUT
DELETE(删除) http://localhost/user DELETE
注意:上述行为是约定方式,约定不是规范,可以打破,所以称Rest风格,而不是Rest规范
5.4.3 Restful开发入门
1 2 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 50 51 52 53 54 55 56 57
| @RestController
@RequestMapping("/user/") public class UserController {
@RequestMapping("/user/{id}") public String restLocation(@PathVariable Integer id){ System.out.println("restful is running ...."); return "success.jsp"; }
@RequestMapping("{id}") public String restLocation2(@PathVariable Integer id){ System.out.println("restful is running ....get:"+id); return "success.jsp"; }
@RequestMapping(value = "{id}",method = RequestMethod.GET) @GetMapping("{id}") public String get(@PathVariable Integer id){ System.out.println("restful is running ....get:"+id); return "success.jsp"; }
@RequestMapping(value = "{id}",method = RequestMethod.POST) @PostMapping("{id}") public String post(@PathVariable Integer id){ System.out.println("restful is running ....post:"+id); return "success.jsp"; }
@RequestMapping(value = "{id}",method = RequestMethod.PUT) @PutMapping("{id}") public String put(@PathVariable Integer id){ System.out.println("restful is running ....put:"+id); return "success.jsp"; }
@RequestMapping(value = "{id}",method = RequestMethod.DELETE) @DeleteMapping("{id}") public String delete(@PathVariable Integer id){ System.out.println("restful is running ....delete:"+id); return "success.jsp"; } }
|
1 2 3 4 5 6 7 8 9
| <filter> <filter-name>HiddenHttpMethodFilter</filter-name> <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class> </filter> <filter-mapping> <filter-name>HiddenHttpMethodFilter</filter-name> <servlet-name>DispatcherServlet</servlet-name> </filter-mapping>
|
开启SpringMVC对Restful风格的访问支持过滤器,即可通过页面表单提交PUT与DELETE请求
页面表单使用隐藏域提交请求类型,参数名称固定为_method,必须配合提交类型method=post使用
1 2 3 4
| <form action="/user/1" method="post"> <input type="hidden" name="_method" value="PUT"/> <input type="submit"/> </form>
|
1 2 3 4 5 6 7 8
| @RestController public class UserController { @RequestMapping(value = "/user/{id}",method = RequestMethod.DELETE) public String restDelete(@PathVariable String id){ System.out.println("restful is running ....delete:"+id); return "success.jsp"; } }
|
5.5 postman工具安装与使用
postman 是 一款可以发送Restful风格请求的工具,方便开发调试。首次运行需要联网注册
