Spring Boot 无侵入式 实现 API 接口统一 JSON 格式返回
blog.csdn.net/qq_34347620/article/details/102239179
无侵入式 统一返回 JSON 格式
其实本没有没打算写这篇博客的,但还是要写一下写这篇博客的起因是因为,现在呆着的这家公司居然没有统一的 API 返回格式?,询问主管他居然告诉我用 HTTP 状态码就够用了(fxxk),天哪 HTTP 状态码真的够用吗?
在仔细的阅读了项目源码后发现,在 API 请求的是居然没有业务异常(黑人问好)。好吧 居然入坑了只能遵照项目风格了,懒得吐槽了。
因为项目已经开发了半年多了, 要是全部接口都做修改工作量还是挺大的, 只能用这种无侵入式的方案来解决.
项目源代码: https://github.com/469753862/galaxy-blogs/tree/master/code/responseResult
定义 JSON 格式
定义返回 JSON 格式
后端返回给前端一般情况下使用 JSON 格式, 定义如下
{
"code": 200,
"message": "OK",
"data": {
}
}
-
code: 返回状态码
-
message: 返回信息的描述
-
data: 返回值
定义 JavaBean 字段
定义状态码枚举类
@ToString
@Getter
public enum ResultStatus {
SUCCESS(HttpStatus.OK, 200, "OK"),
BAD\_REQUEST(HttpStatus.BAD\_REQUEST, 400, "Bad Request"),
INTERNAL\_SERVER\_ERROR(HttpStatus.INTERNAL\_SERVER\_ERROR, 500, "Internal Server Error"),;
/\*\* 返回的HTTP状态码, 符合http请求 \*/
private HttpStatus httpStatus;
/\*\* 业务异常码 \*/
private Integer code;
/\*\* 业务异常信息描述 \*/
private String message;
ResultStatus(HttpStatus httpStatus, Integer code, String message) {
this.httpStatus = httpStatus;
this.code = code;
this.message = message;
}
}
状态码和信息以及 http 状态码就能一一对应了便于维护, 有同学有疑问了为什么要用到 http 状态码呀, 因为我要兼容项目以前的代码, 没有其他原因, 当然其他同学不喜欢 http 状态码的可以吧源码中 HttpStatus 给删除了
定义返回体类
@Getter
@ToString
public class Result<T> {
/\*\* 业务错误码 \*/
private Integer code;
/\*\* 信息描述 \*/
private String message;
/\*\* 返回参数 \*/
private T data;
private Result(ResultStatus resultStatus, T data) {
this.code = resultStatus.getCode();
this.message = resultStatus.getMessage();
this.data = data;
}
/\*\* 业务成功返回业务代码和描述信息 \*/
public static Result<Void> success() {
return new Result<Void>(ResultStatus.SUCCESS, null);
}
/\*\* 业务成功返回业务代码,描述和返回的参数 \*/
public static <T> Result<T> success(T data) {
return new Result<T>(ResultStatus.SUCCESS, data);
}
/\*\* 业务成功返回业务代码,描述和返回的参数 \*/
public static <T> Result<T> success(ResultStatus resultStatus, T data) {
if (resultStatus == null) {
return success(data);
}
return new Result<T>(resultStatus, data);
}
/\*\* 业务异常返回业务代码和描述信息 \*/
public static <T> Result<T> failure() {
return new Result<T>(ResultStatus.INTERNAL\_SERVER\_ERROR, null);
}
/\*\* 业务异常返回业务代码,描述和返回的参数 \*/
public static <T> Result<T> failure(ResultStatus resultStatus) {
return failure(resultStatus, null);
}
/\*\* 业务异常返回业务代码,描述和返回的参数 \*/
public static <T> Result<T> failure(ResultStatus resultStatus, T data) {
if (resultStatus == null) {
return new Result<T>(ResultStatus.INTERNAL\_SERVER\_ERROR, null);
}
return new Result<T>(resultStatus, data);
}
}
因为使用构造方法进行创建对象太麻烦了, 我们使用静态方法来创建对象这样简单明了
Result 实体返回测试
@RestController
@RequestMapping("/hello")
public class HelloController {
private static final HashMap<String, Object> INFO;
static {
INFO = new HashMap<>();
INFO.put("name", "galaxy");
INFO.put("age", "70");
}
@GetMapping("/hello")
public Map<String, Object> hello() {
return INFO;
}
@GetMapping("/result")
@ResponseBody
public Result<Map<String, Object>> helloResult() {
return Result.success(INFO);
}
}
到这里我们已经简单的实现了统一 JSON 格式了, 但是我们也发现了一个问题了, 想要返回统一的 JSON 格式需要返回Result<Object>
才可以, 我明明返回 Object 可以了, 为什么要重复劳动, 有没有解决方法, 当然是有的啦, 下面我们开始优化我们的代码吧
统一返回 JSON 格式进阶 - 全局处理 (@RestControllerAdvice)
我师傅经常告诉我的一句话: “你就是一个小屁孩, 你遇到的问题都已经不知道有多少人遇到过了, 你会想到的问题, 已经有前辈想到过了. 你准备解决的问题, 已经有人把坑填了”。是不是很鸡汤, 是不是很励志, 让我对前辈们充满着崇拜, 事实上他对我说的是: “自己去百度”, 这五个大字, 其实这五个大字已经说明上明的 B 话了, 通过不断的百度和 Google 发现了很多的解决方案.
我们都知道使用 @ResponseBody 注解会把返回 Object 序列化成 JSON 字符串, 就先从这个入手吧, 大致就是在序列化前把 Object 赋值给Result<Object>
就可以了, 大家可以观摩 org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice 和 org.springframework.web.bind.annotation.ResponseBody
@ResponseBody 继承类
我们已经决定从 @ResponseBody 注解入手了就创建一个注解类继承 @ResponseBody, 很干净什么都没有哈哈,@ResponseResultBody 可以标记在类和方法上这样我们就可以跟自由的进行使用了
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE, ElementType.METHOD})
@Documented
@ResponseBody
public @interface ResponseResultBody {
}
ResponseBodyAdvice 继承类
@RestControllerAdvice
public class ResponseResultBodyAdvice implements ResponseBodyAdvice<Object> {
private static final Class<? extends Annotation> ANNOTATION\_TYPE = ResponseResultBody.class;
/\*\*
\* 判断类或者方法是否使用了 @ResponseResultBody
\*/
@Override
public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
return AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ANNOTATION\_TYPE) || returnType.hasMethodAnnotation(ANNOTATION\_TYPE);
}
/\*\*
\* 当类或者方法使用了 @ResponseResultBody 就会调用这个方法
\*/
@Override
public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
// 防止重复包裹的问题出现
if (body instanceof Result) {
return body;
}
return Result.success(body);
}
}
RestControllerAdvice 返回测试
@RestController
@RequestMapping("/helloResult")
@ResponseResultBody
public class HelloResultController {
private static final HashMap<String, Object> INFO;
static {
INFO = new HashMap<String, Object>();
INFO.put("name", "galaxy");
INFO.put("age", "70");
}
@GetMapping("hello")
public HashMap<String, Object> hello() {
return INFO;
}
/\*\* 测试重复包裹 \*/
@GetMapping("result")
public Result<Map<String, Object>> helloResult() {
return Result.success(INFO);
}
@GetMapping("helloError")
public HashMap<String, Object> helloError() throws Exception {
throw new Exception("helloError");
}
@GetMapping("helloMyError")
public HashMap<String, Object> helloMyError() throws Exception {
throw new ResultException();
}
}
是不是很神奇, 直接返回 Object 就可以统一 JSON 格式了, 就不用每个返回都返回Result<T>
对象了, 直接让 SpringMVC 帮助我们进行统一的管理, 简直完美
只想看接口哦, helloError 和 helloMyError 是会直接抛出异常的接口, 我好像没有对异常返回进行统一的处理哦
统一返回 JSON 格式进阶 - 异常处理 (@ExceptionHandler))
卧槽, 异常处理, 差点把这茬给忘了, 这个异常处理就有很多方法了, 先看看我师傅的处理方式, 我刚拿到这个代码的时候很想吐槽, 对异常类的处理这么残暴的吗, 直接用 PrintWriter 直接输出结果, 果然是老师傅, 我要是有 100 个异常类, 不得要写 100 个 if else 了. 赶紧改改睡吧
@Configuration
public class MyExceptionHandler implements HandlerExceptionResolver {
public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response,
Object handler, Exception ex) {
PrintWriter out = getPrintWrite(response);
if (ex instanceof XXXException) {
out.write(JsonUtil.formatJson(ResultEnum.PAY\_ERROR.getCode(), ex.getMessage()));
} else {
out.write(JsonUtil.formatJson(ResultEnum.FAIL.getCode(), "服务器异常"));
}
if (null != out) {
out.close();
}
return mav;
}
private PrintWriter getPrintWrite(HttpServletResponse response) {
PrintWriter out = null;
try {
response.setHeader("Content-type", "text/html;charset=UTF-8");
response.setCharacterEncoding("UTF-8");
out = response.getWriter();
} catch (IOException e) {
log.error("PrintWriter is exception", e);
}
return out;
}
}
上面的代码看看还是没有问题的, 别学过去哦,
搜索 Java 知音公众号,回复 “后端面试”,送你一份 Java 面试题宝典
异常处理 @ResponseStatus(不推荐)
@ResponseStatus 用法如下, 可用在 Controller 类和 Controller 方法上以及 Exception 类上但是这样的工作量还是挺大的
@RestController
@RequestMapping("/error")
@ResponseStatus(value = HttpStatus.INTERNAL\_SERVER\_ERROR, reason = "Java的异常")
public class HelloExceptionController {
private static final HashMap<String, Object> INFO;
static {
INFO = new HashMap<String, Object>();
INFO.put("name", "galaxy");
INFO.put("age", "70");
}
@GetMapping()
public HashMap<String, Object> helloError() throws Exception {
throw new Exception("helloError");
}
@GetMapping("helloJavaError")
@ResponseStatus(value = HttpStatus.INTERNAL\_SERVER\_ERROR, reason = "Java的异常")
public HashMap<String, Object> helloJavaError() throws Exception {
throw new Exception("helloError");
}
@GetMapping("helloMyError")
public HashMap<String, Object> helloMyError() throws Exception {
throw new MyException();
}
}
@ResponseStatus(value = HttpStatus.INTERNAL\_SERVER\_ERROR, reason = "自己定义的异常")
class MyException extends Exception {
}
全局异常处理 @ExceptionHandler(推荐)
把 ResponseResultBodyAdvice 类进行改造一下, 代码有点多了
主要参考了 org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler#handleException() 方法, 有空可以看一下
@Slf4j
@RestControllerAdvice
public class ResponseResultBodyAdvice implements ResponseBodyAdvice<Object> {
private static final Class<? extends Annotation> ANNOTATION\_TYPE = ResponseResultBody.class;
/\*\* 判断类或者方法是否使用了 @ResponseResultBody \*/
@Override
public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
return AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ANNOTATION\_TYPE) || returnType.hasMethodAnnotation(ANNOTATION\_TYPE);
}
/\*\* 当类或者方法使用了 @ResponseResultBody 就会调用这个方法 \*/
@Override
public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
if (body instanceof Result) {
return body;
}
return Result.success(body);
}
/\*\*
\* 提供对标准Spring MVC异常的处理
\*
\* @param ex the target exception
\* @param request the current request
\*/
@ExceptionHandler(Exception.class)
public final ResponseEntity<Result<?>> exceptionHandler(Exception ex, WebRequest request) {
log.error("ExceptionHandler: {}", ex.getMessage());
HttpHeaders headers = new HttpHeaders();
if (ex instanceof ResultException) {
return this.handleResultException((ResultException) ex, headers, request);
}
// TODO: 2019/10/05 galaxy 这里可以自定义其他的异常拦截
return this.handleException(ex, headers, request);
}
/\*\* 对ResultException类返回返回结果的处理 \*/
protected ResponseEntity<Result<?>> handleResultException(ResultException ex, HttpHeaders headers, WebRequest request) {
Result<?> body = Result.failure(ex.getResultStatus());
HttpStatus status = ex.getResultStatus().getHttpStatus();
return this.handleExceptionInternal(ex, body, headers, status, request);
}
/\*\* 异常类的统一处理 \*/
protected ResponseEntity<Result<?>> handleException(Exception ex, HttpHeaders headers, WebRequest request) {
Result<?> body = Result.failure();
HttpStatus status = HttpStatus.INTERNAL\_SERVER\_ERROR;
return this.handleExceptionInternal(ex, body, headers, status, request);
}
/\*\*
\* org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler#handleExceptionInternal(java.lang.Exception, java.lang.Object, org.springframework.http.HttpHeaders, org.springframework.http.HttpStatus, org.springframework.web.context.request.WebRequest)
\* <p>
\* A single place to customize the response body of all exception types.
\* <p>The default implementation sets the {@link WebUtils#ERROR\_EXCEPTION\_ATTRIBUTE}
\* request attribute and creates a {@link ResponseEntity} from the given
\* body, headers, and status.
\*/
protected ResponseEntity<Result<?>> handleExceptionInternal(
Exception ex, Result<?> body, HttpHeaders headers, HttpStatus status, WebRequest request) {
if (HttpStatus.INTERNAL\_SERVER\_ERROR.equals(status)) {
request.setAttribute(WebUtils.ERROR\_EXCEPTION\_ATTRIBUTE, ex, WebRequest.SCOPE\_REQUEST);
}
return new ResponseEntity<>(body, headers, status);
}
}
参考博客列表:
https://www.toutiao.com/i6694404645827117572/ https://blog.csdn.net/qq_36722039/article/details/80825117 http://www.imooc.com/article/260354 https://my.oschina.net/wangkang80/blog/1519189