下面提供4种方式来定制错误数据,最好的一种是第3种。
[TOC]
# 1. SpringBoot默认的错误数据
当没有定制任何错误处理页面和数据时,访问一个不存在的地址 http://localhost:8080/abc SpringBoot根据请求的客户端不同响应不同的数据。
(1)浏览器访问,默认显示html。
![](https://img.kancloud.cn/59/6f/596f86ddfe4e3389e38413031c7359b6_1394x308.jpg)
(2)非浏览器访问,默认显示json。
![](https://img.kancloud.cn/0d/be/0dbe4ce8bff2239e157c9f26f9fc2280_1470x384.jpg)
<br/>
# 2. 自定义异常处理器(一)
无论是浏览器访问,还是非浏览器方法,显示的异常数据都是JSON。
<br/>
步骤如下:
**1. 自定义异常处理器**
```java
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
import java.util.HashMap;
import java.util.Map;
@ControllerAdvice // 标注该类为异常处理器
public class MyExceptionHandler {
@ResponseBody
@ExceptionHandler({RuntimeException.class}) // 要处理的异常
public Map<String, Object> handlerException(Exception e) {
Map<String, Object> map = new HashMap<>();
map.put("code", "我自定义的状态码500000");
map.put("message", "这是我自定义的异常消息哦!");
return map;
}
}
```
**2. controller层**
```java
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;
@Controller
public class IndexController {
@GetMapping("/index")
public String index(@RequestParam("id") int id) {
if (id == 1) {
throw new RuntimeException("RuntimeException!");
}
return "";
}
}
```
**3. 测试**
启动项目后访问一个不存在的地址 http://localhost:8080/index?id=1 ,得到如下结果。
(1)浏览器访问
![](https://img.kancloud.cn/0f/53/0f5345dc4a7971a155825a448a1f3d77_1411x170.jpg)
(2)非浏览器访问
![](https://img.kancloud.cn/f6/15/f615386a09dcd8913c4e56c35d4a8cf8_1483x333.jpg)
<br/>
# 3. 自定义异常处理器(二)
浏览器访问显示错误数据显示为html,非浏览器访问显示为json。
<br/>
步骤如下:
**1. 自定义异常处理器**
```java
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import javax.servlet.http.HttpServletRequest;
import java.util.HashMap;
import java.util.Map;
@ControllerAdvice // 标注该类为异常处理器
public class MyExceptionHandler {
@ExceptionHandler({RuntimeException.class}) // 要处理的异常
public String handlerException(Exception e, HttpServletRequest request) {
Map<String, Object> map = new HashMap<>();
map.put("code", "我自定义的状态码500000");
map.put("message", "这是我自定义的异常消息哦!");
// 为了后面测试,请把status_code设置为500
request.setAttribute("javax.servlet.error.status_code", 500);
request.setAttribute("ext", map);
return "forward:/error"; // 这个error请求由SpringBoot默认提供好了的
}
}
```
**2. 继承DefaultErrorAttributes**
```java
import org.springframework.boot.web.servlet.error.DefaultErrorAttributes;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.WebRequest;
import java.util.Map;
@Component
public class MyErrorAttributes extends DefaultErrorAttributes {
@Override
public Map<String, Object> getErrorAttributes(WebRequest webRequest, boolean includeStackTrace) {
Map<String, Object> map = super.getErrorAttributes(webRequest, includeStackTrace);
map.put("company", "发生异常了,你能拿我怎么办!");
map.put("message", "发生异常了,You get out 吧!");
Map<String, Object> ext = (Map<String, Object>) webRequest.getAttribute("ext", 0);
map.put("ext", ext);
return map;
}
}
```
**3. controller层**
```java
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestParam;
@Controller
public class IndexController {
@GetMapping("/index")
public String index(@RequestParam("id") int id) {
if (id == 1) {
throw new RuntimeException("RuntimeException!");
}
return "";
}
}
```
**4. 错误模板**
*`resources/template/error/5xx.html`*
```html
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
<meta charset="UTF-8">
<title>5xx</title>
</head>
<body>
<div>ext.Message:[[${ext.message}]]</div>
<div>Message:[[${message}]]</div>
<div>code:[[${ext.code}]]</div>
<div>company:[[${company}]]</div>
</body>
</html>
```
**5. 测试**
启动项目后访问 http://localhost:8080/index?id=1 得到如下结果。
(1)浏览器访问
![](https://img.kancloud.cn/bb/95/bb9568355f94ce10c5c7b8b23bea3fd0_1461x251.jpg)
(2)非浏览器访问
```json
{
"timestamp": "2021-07-20T07:50:38.412+0000",
"status": 500,
"error": "Internal Server Error",
"message": "发生异常了,You get out 吧!",
"trace": "java.lang.RuntimeException:
RuntimeException!\r\n\tat
com.example.boot.controller.IndexController.index(IndexController.java:13)\r\n\tat
...
"path": "/index",
"company": "发生异常了,你能拿我怎么办!",
"ext": {
"code": "我自定义的状态码500000",
"message": "这是我自定义的异常消息哦!"
}
}
```
- Mybatis
- mybatis是什么
- mybatis优缺点
- 环境搭建
- 使用步骤
- 传参方式
- 无需传参
- 一个参数
- 多个参数
- 增/删/改
- 查询
- 单表查询
- 一对一查询
- 一对多查询
- 动态SQL
- 注解操作
- Spring
- Spring什么
- Spring优点
- Spring组成
- 第一个Spring程序
- 两大核心技术
- IoC控制反转
- IoC思想
- IoC容器使用步骤
- 属性注入
- IoC注入方式
- 模拟IoC实现
- AOP
- AOP概念
- AOP原理
- AOP关键术语
- AOP编程过程
- 切入点规则
- 5种增强方式
- Spring注解开发
- 注解开发的优势
- Bean注解开发
- AOP注解开发
- 完全注解开发
- 模拟Spring注解开发
- 自动装配
- 配置文件拆分
- SpringBean
- Bean常用属性
- Bean的作用域
- Bean的生命周期
- Spring整合MyBatis
- 整合步骤
- SqlSessionTemplate
- 业务层添加事务
- 事务的作用
- 配置文件事务
- 注解事务
- 事务参数
- SpringMVC
- SpringMVC是什么
- 环境搭建
- 请求流程
- 核心组件
- 前后端交互
- 简单交互演示
- 常用注解
- 后端数据传递至前端
- ServletAPI
- 访问静态资源
- 异常处理
- HandlerExceptionResolver
- 局部异常
- 全局异常
- 转发与重定向
- 转发演示
- 重定向演示
- 转发与重定向的区别
- 获取表单数据
- 表单标签
- REST风格的URL
- 异步处理
- 异步请求
- JSON数据处理
- 中文乱码处理
- 日期处理
- 上传文件
- 拦截器
- 视图解析器
- 视图类型
- 多视图解析器
- 自定义pdf视图
- JSR303数据验证
- JSR303是什么
- 常用约束
- 使用步骤
- SpringMVC整合Mybatis
- 整合步骤
- Mybatis分页插件
- SpringBoot
- SpringBoot是什么
- 环境搭建
- SpringBoot启动分析
- SpringBoot启动类
- 启动过程
- SpringBoot配置文件
- 配置文件类型
- 更改配置文件
- 读取配置文件
- 占位符
- 配置优先级
- 自定义IoC容器
- 定义方式
- 引入Spring配置文件
- @Configuration
- SpringBoot自动配置
- 自动配置原理
- 条件注解
- 自动配置报告
- 自定义自动配置
- 关闭自动配置
- 接管自动配置
- 多环境配置
- CommandLineRunner
- SpringBoot与Web开发
- 引入模板引擎
- Thymeleaf模板
- Freemarker模板
- 静态资源访问
- webjars
- 静态资源位置
- ico图标
- 指定首页
- 更换Web服务器
- 国际化
- 拦截器
- 错误处理机制
- 错误处理机制原理
- 定制错误页面
- 定制错误数据
- 上传文件
- 注册servlet三大组件
- 注册Servlet
- 注册过滤器
- 注册监听器
- 外部Tomcat与jsp模板
- 前后端交互
- 传递json字符串
- 传递js对象
- 传递表单
- 下载功能
- Swagger2文档
- SpringBoot整合JDBC
- 整合步骤
- 核心API
- JdbcTemplate
- 增删改
- 查询
- NamedParameterJdbcTemplate
- 增删改
- 查询
- SpringBoot整合Mybatis
- 整合步骤
- 切换为Druid数据源
- 添加事务
- Mybatis分页插件
- 场景启动器
- 场景启动器是什么
- 自定义场景启动器
- SpringBoot与日志
- 日志框架
- slf4j日志
- slf4j日志实现
- 统一切换为slf4j
- 日志配置
- 日志文件
- 切换日志框架
- 切换日志场景启动器
- SpringBoot与缓存
- JSR107缓存技术
- Spring缓存抽象
- 缓存注解
- SpEL表达式
- 使用缓存
- 自定义key生成器
- 缓存工作原理与流程
- SpringBoot整合Redis
- 整合步骤
- 初步使用
- 序列化机制
- 缓存管理器
- SpringBoot与任务
- 异步任务
- 实现异步任务
- 注意事项与原理
- 自定义线程池
- 定时任务
- cron表达式
- 创建定时任务
- @Scheduled参数
- 动态时间
- 邮件任务
- Quartz定时任务
- Quartz是什么
- 创建定时任务
- 触发器与任务
- 任务的CURD
- 两种触发器
- 并发问题
- 持久化
- 任务持久化
- Quartz集群
- misfire策略
- 打包插件
- appassembler-maven-plugin
- appassembler与assembly配合