Springboot学习06-自定义错误页面完整分析
前言
接着上一篇博客,继续分析Springboot错误页面问题
正文
1-自定义浏览器错误页面(只要将自己的错误页面放在指定的路径下即可)
1-1-Springboot错误页面匹配机制(以404错误为例):
- 1-在模板引擎下:找templates/error/404.html;如果没有,则继续匹配
- 2-在模板引擎下:找templates/error/4XX.html;如果没有,则继续匹配
- 3-在静态资源下:找static/error/404.html;如果没有,则继续匹配
- 4-匹配最后的“围墙”:WhiteLevel Erro Page页面
1-2-补充说明
- 1-模板引擎下的匹配规则;源码分析,请参考本人博客:https://www.cnblogs.com/wobuchifanqie/p/10144151.html
- 2-静态资源下的错误页面,不一定在static路径下,只需要符合静态资源映射规则即可;源码分析,请参考本人博客:https://www.cnblogs.com/wobuchifanqie/p/10112302.html
- 3-WhiteLevel Erro Page页面是动态生成,源码分析,请参考本人博客:https://www.cnblogs.com/wobuchifanqie/p/10144151.html
1-3-demo示例
1-4-简单自定义页面的缺陷
- 1-只能展示Springboot默认的返回信息:timestamp时间戳;status状态;error错误提示;exception异常对象;message异常消息等简单返回信息;无法返回自定义业务数据
2-自定义错误的json
2-1-源码分析
//1-自定义Exception public class DataException extends RuntimeException { public DataException() { super("数据不存在!"); } } //2-自定义handleException方法 @ControllerAdvice public class MyExceptionHandler { @ResponseBody @ExceptionHandler(DataException.class) public Map<String,Object> handleException(Exception e){ Map<String,Object> map = new HashMap<String,Object>(); map.put("code","data error"); map.put("msg",e.getMessage()); return map; } } //3-测试接口 @Controller public class DemoController { @GetMapping(value="test") public String toExceptionPage( ){ throw new DataException(); } }
2-2-页面效果
2-3-缺点:浏览器请求也返回了json数据;不符合期望
3-自定义错误页面,自适应浏览器请求和客户端请求
3-1-源码示例
//1-自定义Exception public class DataException extends RuntimeException { public DataException() { super("数据不存在!"); } } //2-自定义handleException方法 @ControllerAdvice public class MyExceptionHandler { @ExceptionHandler(DataException.class) public String handleException(Exception e, HttpServletRequest request){ //传入我们自己的错误状态码 4xx 5xx,否则就不会进入定制错误页面的解析流程 request.setAttribute("javax.servlet.error.status_code",500); Map<String,Object> map = new HashMap<String,Object>(); map.put("code","data error"); map.put("msg",e.getMessage()); return "forward:/error"; } } //3-测试接口 @Controller public class DemoController { @GetMapping(value="test") public String toExceptionPage( ){ throw new DataException(); } }
3-2-demo示例
3-3-缺点:虽然已经兼容了浏览器请求和客户端请求;但是无法展示业务数据
4-自定义页面终版:自适应浏览器请求和客户端请求,并且允许返回业务数据
4-1-源码解析
//1-自定义ErrorAttributes @Component public class MyErrorAttributes extends DefaultErrorAttributes { //重写getErrorAttributes方法-添加自己的项目数据 public Map<String, Object> getErrorAttributes(WebRequest webRequest, boolean includeStackTrace) { Map<String, Object> errorAttributes = super.getErrorAttributes(webRequest,includeStackTrace); errorAttributes.put("myName","我不吃番茄");//自定义数据 errorAttributes.put("myAge","不告诉你");//自定义数据 return errorAttributes; } } //2-自定义Exception public class DataException extends RuntimeException { public DataException() { super("数据不存在!"); } } //3-自定义handleException方法 @ControllerAdvice public class MyExceptionHandler { @ExceptionHandler(DataException.class) public String handleException(Exception e, HttpServletRequest request){ //传入我们自己的错误状态码 4xx 5xx,否则就不会进入定制错误页面的解析流程 request.setAttribute("javax.servlet.error.status_code",500);//这里只接受500状态错误 Map<String,Object> map = new HashMap<String,Object>(); map.put("code","data error"); map.put("msg",e.getMessage()); request.setAttribute("extra", map);//放在request中的数据,在前端页面中都可以取出来 return "forward:/error";//并不直接返回视图名称或json数据,请求转发到"/error",让Springboo按流程处理处理,从而达到自适应浏览器请求和客户端请求; } } //4-测试接口 @Controller public class DemoController { @GetMapping(value="test") public String toExceptionPage( ){ throw new DataException();//主动抛出一个500错误,用于测试 } }
//templates/error/5XX.html <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <h1>自定义页面 --路径:templates/error/5XX.html页面 --优先级别:2</h1> <h1>status:[[${status}]]</h1> <h2>timestamp:[[${timestamp}]]</h2> <h2>exception:[[${exception}]]</h2> <h2>myName:[[${myName}]]</h2> <h2>myAge:[[${myAge}]]</h2> <h2>extra-code:[[${extra.code}]]</h2> <h2>extra-msg:[[${extra.msg}]]</h2> </body> </html>