springboot默认的错误处理机制:
浏览器:返回一个错误页面:
data:image/s3,"s3://crabby-images/bb48d/bb48de7fc9b2295b92ceb816b033f75c0eda1e9a" alt=""
其他客户端:返回一个Json数据
data:image/s3,"s3://crabby-images/d1388/d13884185abb66cd0859d9eecf1f94d611eaf88f" alt=""
原理:可以参考ErrorMvcAutoConfiguration类
DefaultErrorAttributes:错误页面显示的信息
BasicErrorController:处理默认/error请求
@Controller
@RequestMapping("${server.error.path:${error.path:/error}}")
public class BasicErrorController extends AbstractErrorController {
//产生html错误信息,浏览器发送的错误请求在这里处理
@RequestMapping(produces = MediaType.TEXT_HTML_VALUE)
public ModelAndView errorHtml(HttpServletRequest request, HttpServletResponse response) {
HttpStatus status = getStatus(request);
Map<String, Object> model = Collections
.unmodifiableMap(getErrorAttributes(request, isIncludeStackTrace(request, MediaType.TEXT_HTML)));
response.setStatus(status.value());
ModelAndView modelAndView = resolveErrorView(request, response, status, model);
return (modelAndView != null) ? modelAndView : new ModelAndView("error", model);
}
//产生Json错误信息,其他客户端发送的错误请求在这里处理
@RequestMapping
public ResponseEntity<Map<String, Object>> error(HttpServletRequest request) {
HttpStatus status = getStatus(request);
if (status == HttpStatus.NO_CONTENT) {
return new ResponseEntity<>(status);
}
Map<String, Object> body = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.ALL));
return new ResponseEntity<>(body, status);
}
}
根据不同客户端请求头信息区别响应格式:
data:image/s3,"s3://crabby-images/4f42b/4f42ba04863b80c9cd9ec6363ef645ac3ad3768b" alt=""
data:image/s3,"s3://crabby-images/c4594/c4594bdd647b6758094dbce3fbf57d68d4affd45" alt=""
ErrorPageCustomizer:定制错误响应规则
data:image/s3,"s3://crabby-images/ebcca/ebcca10edfd5bb072162a0914b334a3b59a2b012" alt=""
data:image/s3,"s3://crabby-images/7aa48/7aa48db5eb7b707007f158021ae3dbc8c0344941" alt=""
data:image/s3,"s3://crabby-images/bb6da/bb6da38da2b123878f78212dc2358ce85369dc63" alt=""
系统出现错误以后,默认响应规则为跳转到/error请求进行处理
DefaultErrorViewResolver:去哪个响应页面
@Override
public ModelAndView resolveErrorView(HttpServletRequest request, HttpStatus status, Map<String, Object> model) {
//所有的ErrorViewResolver得到ModelAndView
ModelAndView modelAndView = resolve(String.valueOf(status.value()), model);
if (modelAndView == null && SERIES_VIEWS.containsKey(status.series())) {
modelAndView = resolve(SERIES_VIEWS.get(status.series()), model);
}
return modelAndView;
}
步骤:
一但系统出现4xx或5xx的错误,ErrorPageCustomizer就会生效,来到/error请求,被BasicErrorController处理,去哪个响应页面由DefaultErrorViewResolver处理,错误信息由DefaultErrorAttributes解析
目的:定制错误响应
1)、如何定制错误页面?
有模板引擎的情况下,将错误页面以状态码命名,放在模板引擎文件夹下的error文件夹下,发生此状态码错误就会来到这个页面,也可以使用4xx或5xx来命名,当没有精确的错误状态码匹配时,就会根据错误类型访问对应的4xx或5xx页面,原则是精确优先,有精确的先访问精确的状态码.html,
data:image/s3,"s3://crabby-images/7a555/7a5552829119edf6363cae7833a846cb09fb9cca" alt=""
data:image/s3,"s3://crabby-images/4f46b/4f46ba151b16ff5b67bd0430d24f3cc75a3475bc" alt=""
data:image/s3,"s3://crabby-images/ccf2b/ccf2bcb36fe0f65253d72bb2c15418c7ccd9b844" alt=""
错误页面可以获取的信息:
timestamp:时间戳
status:状态码
error:错误提示
exception:异常信息
message:错误信息
errors:JSR303数据校验错误信息
在自定义错误页面获取错误信息:
data:image/s3,"s3://crabby-images/a2554/a2554a6b90cf44782dcce01fdc342763a7b320a4" alt=""
data:image/s3,"s3://crabby-images/6053e/6053ee33a224a7c23f5469bdb8d4f34355cd960e" alt=""
没有模板引擎默认在静态资源文件夹下寻找错误页面,命名规则一致
但是没有模板引擎则无法获取错误信息
2)、如何定制错误Json数据?
默认Json格式数据
data:image/s3,"s3://crabby-images/848da/848dacc08ee333dd1733993b13c5b6c44f824a33" alt=""
创建自定义异常
data:image/s3,"s3://crabby-images/b2f74/b2f741a080258036f517f1159b803cf97dcdf93e" alt=""
data:image/s3,"s3://crabby-images/e9651/e9651327db1927714c6b814176d0d3419a3e860d" alt=""
修改显示的数据
@ControllerAdvice//指明这是一个异常处理器
public class myExceptionHandler {
@ResponseBody//返回Json数据
@ExceptionHandler(userNotFoundException.class)//标注处理什么异常
public Map<String,Object> handler(Exception e){
Map<String,Object> map = new HashMap<>();
map.put("code","user.notFound");//处理异常返回的信息
map.put("message",e.getMessage());
return map;//返回map生成的Json数据
}
}
data:image/s3,"s3://crabby-images/e08d2/e08d2378757cd0dd2994f1d603ae8ad291169315" alt=""
问题:此时不论浏览器还是其他客户端,访问该异常都是Json格式
data:image/s3,"s3://crabby-images/879a4/879a4ae32249225178ca97297604efda649fd6d4" alt=""
解决:设置自适应
@ExceptionHandler(userNotFoundException.class)
public String handler(Exception e){
Map<String,Object> map = new HashMap<>();
map.put("code","user.notFound");
map.put("message",e.getMessage());
return "forward:/error";//转发到/error请求达到自适应效果
}
data:image/s3,"s3://crabby-images/acdd6/acdd69536df31cba6782989c1f2a0c2251130c47" alt=""
问题:不是我们自定义的页面了,且状态码为200,这是分明是正确访问的结果
解决:在自定义异常处理器时,设置异常的状态码
data:image/s3,"s3://crabby-images/77a9c/77a9ca29a6c632d60438ea1e72d0cd3311574a65" alt=""
data:image/s3,"s3://crabby-images/c9707/c97079934e148d92f37596115540f5593d0e82f7" alt=""
问题:自定义的信息无法显示
data:image/s3,"s3://crabby-images/3f337/3f33749809af72a408c004bb20244462f86f5885" alt=""
解决:自定义错误属性
data:image/s3,"s3://crabby-images/dda15/dda15891418bd08e67e5b2cd3750518daf0437b8" alt=""
创建自定义错误属性类,继承DefaultErrorAttributes
@Component
public class myErrorAttributs extends DefaultErrorAttributes {
@Override
public Map<String, Object> getErrorAttributes(WebRequest webRequest, boolean includeStackTrace) {
//使用getErrorAttributes获取的map集合就是转换为错误Json信息的map集合
Map<String, Object> map = super.getErrorAttributes(webRequest, includeStackTrace);
//往map集合内添加新的键值对,就能添加Json的信息
map.put("myInfo","cn.yzx");
//或者传入同名key就可以覆盖已有的信息
map.put("status",499);
//也可以从各种域中获取共享的信息,0表示request域
Map<String, Object> ext = (Map<String, Object>) webRequest.getAttribute("ext", 0);
//添加到map中即可
map.put("ext",ext);
return map;
}
}
data:image/s3,"s3://crabby-images/a9391/a939126e6f81f7b31c68507b55c548456ce09f02" alt=""
网友评论