因为我们浏览器可以自己去写错误页面,其他客户端的话返回的是json数据,我们应该怎么自定义去修改这个json数据,换成我们发送的
1、第一种方式,利用@ControllerAdvice注解,来创建一个处理全局异常的Controller类
使用@ControllerAdvice可以实现三个方面的功能:
1、全局异常处理 |
---|
2、全局数据绑定 |
3、全局数据预处理 |
这是 SpringMVC 提供的功能,在 Spring Boot 中可以直接使用
@ControllerAdvice注解的第一个功能: 全局异常处理
如果需要进行全局异常处理,我们需要创建一个类,在类上添加这个注解
//这个注解专门用于异常处理类上
@ControllerAdvice
//创建自定义的异常处理器
public class MyExceptionHandler {
//该用来捕获异常
//1、一个Controller下多个@ExceptionHandler上的异常类型不能出现一样的,否则运行时抛异常
//2、方法参数是某个异常类的class,代表这个方法专门处理该类异常
//3、采取就近原则,如果异常类有父类,父类还有父类,归父类处理
//这样出现异常的时候,就可以自己封装数据返回
@ExceptionHandler(Exception.class)
@ResponseBody //返回json数据
public HashMap<String, Object> handlerException(Exception e){//可以获取到捕获的异常对象
HashMap<String, Object> map = new HashMap<>();
map.put("code","user.notexist");
map.put("message",e.getMessage());
return map;
}
}
@ControllerAdvice注解的第二个功能: 全局数据绑定
我们可以将一些公共的数据定义在添加了 @ControllerAdvice 注解的类中,
这样,在每一个 Controller 的接口中,就都能够访问导致这些数据
@ControllerAdvice
public class MyGlobalExceptionHandler {
//这个注解标注的返回的数据是一个全局数据,默认情况下,这个全局数据的 key 就是返回的变量名,value 就是方法返回值,
//当然开发者可以通过 @ModelAttribute 注解的 name 属性去重新指定 key。
//定义完成后,在任何一个Controller 的接口中,都可以获取到这里定义的数据
//如果方法上使用了 @ExceptionHandler和@ResponseBody的时候,异常返回的数据就不会返回给页面了
@ModelAttribute("test")
public HashMap<String, Object> handlerException(Exception e){
HashMap<String, Object> map = new HashMap<>();
map.put("code","user.notexist");
map.put("message",e.getMessage());
System.out.println("1111111111111");
return map;
}
测试结果:{exception=java.lang.Exception, org.springframework.validation.BindingResult.exception=org.springframework.validation.BeanPropertyBindingResult:
0 errors, test={code=user.notexist, message=null}} //获取到了返回的数据
@ControllerAdvice注解的第三个功能: 全局数据预处理
//在传递参数的时候,如果Book类和Author 都有相同的属性,前台传递无法区分,可以使用@ControllerAdvice 的全局数据预处理解决这个问题
@PostMapping("/book")
public void addBook(Book book, Author author) {
System.out.println(book);
System.out.println(author);
}
解决步骤如下:
1.给接口中的变量取别名
@PostMapping("/book")
public void addBook(@ModelAttribute("b") Book book, @ModelAttribute("a") Author author) {
System.out.println(book);
System.out.println(author);
}
2.进行请求数据预处理
@InitBinder("b")
public void b(WebDataBinder binder) {
binder.setFieldDefaultPrefix("b.");
}
@InitBinder("a")
public void a(WebDataBinder binder) {
binder.setFieldDefaultPrefix("a.");
}
@InitBinder("b") 注解表示该方法用来处理和Book和相关的参数,在方法中,给参数添加一个 b 前缀,即请求参数要有b前缀.
3.发送请求
请求发送时,通过给不同对象的参数添加不同的前缀,可以实现参数的区分.