点击上方蓝色“方志朋”,选择“设为星标”

回复“666”获取独家整理的学习资料!

无侵入式 统一返回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")@ResponseBodypublic 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*/@Overridepublic boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {return AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ANNOTATION_TYPE) || returnType.hasMethodAnnotation(ANNOTATION_TYPE);}/*** 当类或者方法使用了 @ResponseResultBody 就会调用这个方法*/@Overridepublic 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;}
}

上面的代码看看还是没有问题的, 别学过去哦,

异常处理@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 */@Overridepublic boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {return AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ANNOTATION_TYPE) || returnType.hasMethodAnnotation(ANNOTATION_TYPE);}/** 当类或者方法使用了 @ResponseResultBody 就会调用这个方法 */@Overridepublic 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);}
}
热门内容:
  • MySQL大表优化方案

  • 从零搭建 Spring Cloud 服务(超级详细)

  • 编写 if 时不带 else,你的代码会更好!

  • 第 3 次读 Effective Java,这 58 个技巧最值!

  • 10大黑客专用的 Linux 操作系统,每个都很酷!

最近面试BAT,整理一份面试资料《Java面试BAT通关手册》,覆盖了Java核心技术、JVM、Java并发、SSM、微服务、数据库、数据结构等等。获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。
明天见(。・ω・。)ノ♡

Spring Boot 无侵入式 实现 API 接口统一 JSON 格式返回相关推荐

  1. Spring Boot 无侵入式 实现API接口统一JSON格式返回

    作者 | 小魏小魏我们去那里呀 来源 | blog.csdn.net/qq_34347620/article/details/102239179 无侵入式 统一返回JSON格式 其实本没有没打算写这篇 ...

  2. java restful接口开发实例_实战:基于Spring Boot快速开发RESTful风格API接口

    写在前面的话 这篇文章计划是在过年期间完成的,示例代码都写好了,结果亲戚来我家做客,文章没来得及写.已经很久没有更新文章了,小伙伴们,有没有想我啊.言归正传,下面开始,今天的话题. 目标 写一套符合规 ...

  3. matlab api接口调用json格式

    matlab 的api接口调用有2个函数,一个是webwrite,一个是webread,webwrite类似post,将数据写入 RESTful Web 服务:webread类似get,从 RESTf ...

  4. Spring Boot入门(16):Spring Boot 整合 Swagger-UI 实现在线API接口文档 | 超级详细,建议收藏

    1. 前言

  5. 解决Spring Boot 整合Security后,所有接口提示Unauthorized、返回401

    解决办法 在启动类前的@SpringBootApplication注解中加入exclude属性SecurityAutoConfiguration和ManagementWebSecurityAutoCo ...

  6. 中央气象台 API接口(json格式)

    1. 数据接口 http://www.weather.com.cn/data/sk/101010100.html http://www.weather.com.cn/data/cityinfo/101 ...

  7. java+swagger+侵入_Spring boot+Swagger配置无侵入式Restful接口(一)

    最近一直遇到有人问我spring boot +swagger怎么配置无侵入式的restful接口,这段时间一直很忙,没有时间,刚好星期六,趁着休息时间给大家写个demo,在这里我就粘贴一些步骤,具体的 ...

  8. spring学习笔记(14)引介增强详解:定时器实例:无侵入式动态增强类功能

    引介增强实例需求 在前面我们已经提到了前置.后置.环绕.最终.异常等增强形式,它们的增强对象都是针对方法级别的,而引介增强,则是对类级别的增强,我们可以通过引介增强为目标类添加新的属性和方法,更为诱人 ...

  9. Java spring boot 阿里云调用人脸识别接口,本地sdk上传到阿里云调用api

    Java spring boot 阿里云调用人脸识别接口 没有写测试类,工具类如下,有access_key_id和access_key_secret传参调用就可使用 代码如下: pom.xml依赖 & ...

最新文章

  1. jquery 插件开发的作用域及基础
  2. comboBox设置为只读(只选)
  3. 企业网络安全之安全维度
  4. 网上看的一篇文章,感觉会给程序员一些启发
  5. 二分图模板(女生赛要用)
  6. OpenCV bgfg分割的实例(附完整代码)
  7. django设置超级管理员_Django的简介与安装
  8. html边框自动变颜色,css怎么设置边框颜色?
  9. android 过滤emoji表情符号,android怎样过滤字符串中的emoji表情
  10. 加快读博失败的10种方法
  11. vector向量容器元素排序与查找
  12. jQuery.fn.extend()方法
  13. 软件需求与分析——大二下需会知识点
  14. 惠普136a芯片清零_怎么把惠普打印机芯片清零
  15. iOS13深色模式/暗黑模式导航栏不自动适配的解决方案
  16. Unity插件Odin入门
  17. php 搜索多模型,基于迅搜(xunsearch) + Laravel Scout 实现 Laravel 学院全文搜索功能(支持多模型搜索)...
  18. 对工作节点执行drain操作时,通过pdb保护pod副本数
  19. html中span怎么写,html的span标签怎么使用
  20. 灰度持续加仓,FIL即将大涨

热门文章

  1. 人工神经网络:感知器
  2. 学习进度条(第一周)
  3. maven生命周期理解
  4. 学习总结--团队项目
  5. 读书笔记--C陷阱与缺陷(三)
  6. Javascript刷题 》 查找数组元素位置
  7. Eclipse用法和技巧十:显示代码outline
  8. StringBuilder、StringBuffer、String区别
  9. html标签(2)--有序列表与无序列表
  10. 【组队学习】【30期】李宏毅机器学习(含深度学习)