当前位置:首页 > 科技  > 软件

优雅地统一处理接口返回值的最佳实践

来源: 责编: 时间:2023-12-05 09:24:45 324观看
导读1. 介绍如果你正在寻找一种方法来规范化你的RESTful API的返回值,那么这篇文章将是你的理想选择。通过阅读这篇文章,你将了解到如何使用Spring ResponseBodyAdvice来改善你的API设计和实现。为什么要统一接口返回值?统一

1. 介绍

如果你正在寻找一种方法来规范化你的RESTful API的返回值,那么这篇文章将是你的理想选择。通过阅读这篇文章,你将了解到如何使用Spring ResponseBodyAdvice来改善你的API设计和实现。dWf28资讯网——每日最新资讯28at.com

为什么要统一接口返回值?dWf28资讯网——每日最新资讯28at.com

  • 统一规范:通过 ResponseBodyAdvice 可以对所有接口返回值进行统一的包装,从而制定统一的规范,使得接口返回值更加清晰、易于理解。这对于整个应用程序的接口管理非常有帮助,可以避免不同接口返回值格式不一致的问题。
  • 可读性增强:通过包装返回值,可以添加必要的字段,如状态码、状态消息等,使得接口返回值更加易于阅读和理解。这对于开发者和用户来说都是非常有益的,可以更方便地了解接口的返回结果。
  • 扩展性:通过 ResponseBodyAdvice 可以很方便地扩展接口返回值的格式,例如添加JSON格式的返回值,而不需要修改原有的接口代码。这使得应用程序具有更好的扩展性,可以根据需求灵活地添加新的返回值格式。

dWf28资讯网——每日最新资讯28at.com

2. 开发流程

定义统一返回值的包装类dWf28资讯网——每日最新资讯28at.com

public class R {  private Integer code ;  private Object data ;  private String message ;  public R(Integer code, Object data, String message) {    this.code = code ;    this.data = data ;    this.message = message ;  }  public static R success(Object data) {    return new R(200, data, "success") ;  }  public static R failure(String message) {    return new R(500, null, message) ;  }}

自定义ResponseBodyAdvicedWf28资讯网——每日最新资讯28at.com

@RestControllerAdvicepublic class PackResponseBodyAdvice implements ResponseBodyAdvice<Object> {  @Resource  private ObjectMapper objectMapper ;  @Override  public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {    // 只有返回值不是R类型的时候才通过该Advice进行处理    return !returnType.getParameterType().equals(R.class) ;  }  @Override  public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType,      Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request,      ServerHttpResponse response) {    // 统一返回值处理    return R.success(body) ;  }}

测试接口dWf28资讯网——每日最新资讯28at.com

@RestController@RequestMapping("/advices")public class AdviceController {  @GetMapping("/str")  public String str() {    return "success" ;  }  @GetMapping("/{id}")  public User body(@PathVariable("id") Long id) {    return new User(id, "张三 - " + new Random().nextInt(1000)) ;  }}

首先,测试接口/advices/{id}dWf28资讯网——每日最新资讯28at.com

图片图片dWf28资讯网——每日最新资讯28at.com

处理了最终的返回结果。dWf28资讯网——每日最新资讯28at.com

继续测试/advices/strdWf28资讯网——每日最新资讯28at.com

图片图片dWf28资讯网——每日最新资讯28at.com

程序出错了dWf28资讯网——每日最新资讯28at.com

控制台输出dWf28资讯网——每日最新资讯28at.com

java.lang.ClassCastException: class com.pack.common.dto.R cannot be cast to class java.lang.String (com.pack.common.dto.R is in unnamed module of loader 'app'; java.lang.String is in module java.base of loader 'bootstrap')  at org.springframework.http.converter.StringHttpMessageConverter.addDefaultHeaders(StringHttpMessageConverter.java:44) ~[spring-web-5.3.27.jar:5.3.27]  at org.springframework.http.converter.AbstractHttpMessageConverter.write(AbstractHttpMessageConverter.java:211) ~[spring-web-5.3.27.jar:5.3.27]

出现ClassCastException错误,是由于Controller接口返回值是String,那么匹配到的HttpMessageConverter是StringHttpMessageConverter处理,而该转换器的调用是在ResponseBodyAdvice之后执行,这时候的字符串已经被转换成了R对象,所以最后在write时就出现了类型转换错误。通过如下方式处理dWf28资讯网——每日最新资讯28at.com

public Object beforeBodyWrite(Object body,       MethodParameter returnType,       MediaType selectedContentType,      Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request,      ServerHttpResponse response) {  if (body instanceof String) {    try {      return this.objectMapper.writeValueAsString(R.success(body)) ;    } catch (JsonProcessingException e) {      e.printStackTrace();    }  }    return R.success(body) ;}

图片dWf28资讯网——每日最新资讯28at.com

针对返回值是String类型的正常了。dWf28资讯网——每日最新资讯28at.com

通过自定义注解排除那些不需要处理的接口dWf28资讯网——每日最新资讯28at.com

@Target({ElementType.METHOD, ElementType.TYPE})@Retention(RetentionPolicy.RUNTIME)public @interface NoR {}

修改PackResponseBodyAdvice#supports方法,添加NoR注解的判断dWf28资讯网——每日最新资讯28at.com

public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {  // 方法上或者是类上没有NoR注解  return (!returnType.hasMethodAnnotation(NoR.class)             || AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), NoR.class))          && !returnType.getParameterType().equals(R.class)  ;}

这样就可以控制具体哪些方法不进行处理了。dWf28资讯网——每日最新资讯28at.com

通过自定义ResponseBodyAdvice,我们可以实现接口统一返回值的处理,从而提高了接口的可读性、扩展性和错误处理能力。同时,这也有助于保持代码的清晰和规范。通过学习和实践,我们可以更好地利用SpringMVC相应的功能,开发出更优秀的应用程序。dWf28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-38112-0.html优雅地统一处理接口返回值的最佳实践

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 一次 K8s 升级,竟然导致滴滴故障 12 小时?

下一篇: Python中的单例模式,这些你都会吗?

标签:
  • 热门焦点
  • 俄罗斯:将审查iPhone等外国公司设备 保数据安全

    iPhone和特斯拉都属于在各自领域领头羊的品牌,推出的产品也也都是数一数二的,但对于一些国家而言,它们的产品可靠性和安全性还是在限制范围内。近日,俄罗斯联邦通信、信息技术
  • 小米官宣:2023年上半年出货量中国第一!

    今日早间,小米电视官方微博带来消息,称2023年小米电视上半年出货量达到了中国第一,同时还表示小米电视的巨屏风暴即将开始。“公布一个好消息2023年#小米电视上半年出货量中国
  • 6月iOS设备好评榜:第一蝉联榜首近一年

    作为安兔兔各种榜单里变化最小的那个,2023年6月的iOS好评榜和上个月相比没有任何排名上的变化,仅仅是部分设备好评率的下降,长年累月的用户评价和逐渐退出市场的老款机器让这
  • 一加首款折叠屏!一加Open渲染图出炉:罕见单手可握小尺寸

    8月5日消息,此前就有爆料称,一加首款折叠屏手机将会在第三季度上市,如今随着时间临近,新机的各种消息也开始浮出水面。据悉,这款新机将会被命名为&ldquo;On
  • 每天一道面试题-CPU伪共享

    前言:了不起:又到了每天一到面试题的时候了!学弟,最近学习的怎么样啊 了不起学弟:最近学习的还不错,每天都在学习,每天都在进步! 了不起:那你最近学习的什么呢? 了不起学弟:最近在学习C
  • 共享单车的故事讲到哪了?

    来源丨海克财经与共享充电宝相差不多,共享单车已很久没有被国内热点新闻关照到了。除了一再涨价和用户直呼用不起了。近日多家媒体再发报道称,成都、天津、郑州等地多个共享单
  • 大厂卷向扁平化

    来源:新熵作者丨南枝 编辑丨月见大厂职级不香了。俗话说,兵无常势,水无常形,互联网企业调整职级体系并不稀奇。7月13日,淘宝天猫集团启动了近年来最大的人力制度改革,目前已形成一
  • 东方甄选单飞:有些鸟注定是关不住的

    文/彭宽鸿编辑/罗卿东方甄选创始人俞敏洪带队的&ldquo;7天甘肃行&rdquo;直播活动已在近日顺利收官。成立后一年多时间里,东方甄选要脱离抖音自立门户的传闻不绝于耳,&ldquo;7
  • OPPO Reno10 Pro英雄联盟定制礼盒公布:萨勒芬妮同款配色梦幻十足

    5月24日,OPPO推出了全新的OPPO Reno 10系列,包含OPPO Reno10、OPPO Reno10 Pro和OPPO Reno10 Pro+三款新机,全系标配了超光影长焦镜头,是迄今为止拍照
Top