0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看威廉希尔官方网站 视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

让代码变得更简洁的小技巧

科技绿洲 来源:Java威廉希尔官方网站 指北 作者:Java威廉希尔官方网站 指北 2023-10-08 14:46 次阅读

目前很多项目都是前后端分离,前后端会事先约定好返回格式。那么后端如何做,才能优雅的返回统一格式呢,接下来,请大家跟着我,一步步来实现。

1. 直接返回结果

先看一下最基本的例子,直接将结果原封不动返回:

@Data
@AllArgsConstructor
@JsonIgnoreProperties(ignoreUnknown = true)
public class TestVo {

    private static final long serialVersionUID = 1L;

    @Schema(name = "姓名")
    private String name;

    @Schema(name = "年龄")
    private Integer age;

}
@RestController
@RequestMapping(value = "/test")
public class TestApi {

    @GetMapping("/simple")
    public TestVo simple() {
        TestVo testVo = new TestVo("张三", 30);
        return testVo;
    }
}

返回结果:

{
    "name": "张三",
    "age": 30
}

2. 约定返回格式

假如已经与前端开发妹子约定好了格式,比如:

{
    "code": 0,
    "msg": "错误信息",
    "data": 实际返回结果
}

那么我们首先需要编写一个封装结果类Result。为了方便封装,在这个类中增加一个success方法:

@Data
@JsonInclude(JsonInclude.Include.NON_NULL)
public class Result< T > implements Serializable {

    private static final long serialVersionUID = 1L;

    /**
     * 返回编码
     */
    private Integer code;

    /**
     * 编码描述
     */
    private String msg;

    /**
     * 业务数据
     */
    private T data;

    /**
     * 返回成功结果对象
     *
     * @param data
     * @param < T >
     * @return
     */
    public static < T > Result< T > success(T data) {
        Result result = new Result();
        result.setCode(0);
        result.setMsg("success");
        result.setData(data);
        return result;
    }
}

3. 返回统一格式结果

后台接口代码微调一下,返回值改为Result,泛型为原返回值的类型:

@RestController
@RequestMapping(value = "/test")
public class TestApi {

    @GetMapping("/withResult")
    public Result< TestVo > withResult() {
        TestVo testVo = new TestVo("张三", 30);
        return Result.success(testVo);
    }
}

返回结果:

{
    "code": 0,
    "msg": "success",
    "data": {
        "name": "张三",
        "age": 30
    }
}

至此,返回结果完美符合格式。

但是这样的代码并不算简洁:每个接口的返回值都必须是Result<>,并且return的时候都要用Result.success()方法封装一下。

那么,有没有更优雅的方法?我们继续往下看:

4. 切片封装统一格式

  1. 编写注解

实际使用场景中,并不是所有接口都需要统一格式。我们这里使用一个注解作为开关,按需控制接口返回格式。

@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface ApiResult {
    String value() default "";

    int successCode() default 0;

    String successMsg() default "success";

    Class< ? extends IResult > resultClass() default Result.class;
}
  1. 编写ControllerAdvice
@ControllerAdvice
public class MyResponseBodyAdvice implements ResponseBodyAdvice {

    protected boolean isStringConverter(Class converterType) {
        return converterType.equals(StringHttpMessageConverter.class);
    }

    protected boolean isApiResult(MethodParameter returnType) {
        return returnType.hasMethodAnnotation(ApiResult.class);
    }

    @Override
    public boolean supports(MethodParameter returnType, Class converterType) {
        return !isStringConverter(converterType) && isApiResult(returnType);
    }

    @Override
    public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType,
                                Class selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
        //关键                         
        return Result.success(body);
    }

}

这里有一点要 注意 ,这个advice中supports方法中判断返回结果类型必须为非String类型。如果返回结果为String类型,那么result要转为json字符串后再返回,需要再写一个advice。

  1. 见证奇迹的时刻到了
@ApiResult
@GetMapping("/withResultHide")
public TestVo withResultHide() {
    TestVo testVo = new TestVo("张三", 30);
    return testVo;
}

这段代码与最开始一样,并没有返回Result,仅仅加上了@ApiResult注解,我们看返回结果:

{
    "code": 0,
    "msg": "success",
    "data": {
        "name": "张三",
        "age": 30
    }
}

大功告成!

以上只是最精简的例子,实际使用中还结合了 统一异常封装、自定义返回格式 等功能。我们注意到@ApiResult注解中,有三个参数:successCode、successMsg、resultClass,就是为了自定义返回格式预留的,下面再看两个场景:

5. 自定义返回格式

场景1:返回成功时code为200

如果个别接口的返回格式与默认格式相同,但是要求code等于200时才代表成功,那么修改下successCode参数即可:

@ApiResult(successCode = 200, successMsg = "ok")
@GetMapping("/withResultHide")
public TestVo withResultHide() {
    TestVo testVo = new TestVo("张三", 30);
    return testVo;
}

返回成功时,结果中的code和msg都变为设置的值:

{
    "code": 200,
    "msg": "ok",
    "data": {
        "name": "张三",
        "age": 30
    }
}
场景2:自定义返回格式

如果某个接口的返回格式不是默认的返回格式,比如约定返回returnCode、returnDesc、data(对应默认的code、msg、data)。那么则需要新增一个返回结果类,比如ReturnResult:

@Data
@JsonInclude(JsonInclude.Include.NON_NULL)
public class ReturnResult< T > implements Serializable {

    private static final long serialVersionUID = 1L;

    /**
     * 返回编码
     */
    private String returnCode;

    /**
     * 编码描述
     */
    private String returnDesc;

    /**
     * 业务数据
     */
    private T data;

    /**
     * 返回成功结果对象
     *
     * @param data
     * @param < T >
     * @return
     */
    public static < T > ReturnResult< T > success(T data) {
        ReturnResult result = new ReturnResult();
        result.setReturnCode(0);
        result.setReturnDesc("success");
        result.setData(data);
        return result;
    }
}

然后修改接口上的@ApiResult注解中的resultClass属性

@ApiResult(resultClass = ReturnResult.class)
@GetMapping("/withResultHide")
public TestVo withResultHide() {
    TestVo testVo = new TestVo("张三", 30);
    return testVo;
}

这时,返回结果就变为想要的格式了:

{
    "returnCode": "0",
    "returnDesc": "success",
    "data": {
        "name": "张三",
        "age": 30
    }
}

小结

只要按照上面一步步改造,即可实现统一返回格式,既简洁、又优雅。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 接口
    +关注

    关注

    33

    文章

    8611

    浏览量

    151247
  • 代码
    +关注

    关注

    30

    文章

    4790

    浏览量

    68653
收藏 人收藏

    评论

    相关推荐

    【公测】新版文档中心上线,开发变得简单

    11月3日,机智云文档中心全新改版,新增和优化了 “X” 功能,快速掌握物联网开发威廉希尔官方网站 变得非常简单。1、新增导航栏。结构不明,不知道什么是云端、APP开发、设备端联网开发在哪?新增导航栏,一目了然
    发表于 11-07 14:27

    传感器变得简单

    的传感器,下一步就是为传感器编写应用程序代码。并非直接跳到微控制器的代码,而是在Python编写应用程序代码。许多总线 工具在编写脚本(writing scripts)配置了插件(plug-in)和范例码
    发表于 11-07 15:24

    智能化电能计量威廉希尔官方网站 如何世界变得环保

    智能化电能计量威廉希尔官方网站 提升能效,世界变得环保
    发表于 02-01 07:49

    单片机编程中怎么效率提高,代码简洁

    单片机编程中怎么效率提高,代码简洁
    发表于 10-20 06:19

    怎么我笔记本电池变得持久?

    怎么我笔记本电池变得持久? 1.在使用笔记本电脑之前认真阅读使用手册的电池保养部分;   2.第一次充电时,你应该连续地把电池
    发表于 01-20 14:12 587次阅读

    如何编写高效简洁的C语言代码

      引言:   编写高效简洁的C语言代码,是许多软件工程师追求的目标。本文就工作中的一些体会和经验做相关的阐述,不对的地方请各位指教。   第1招:以
    发表于 08-19 14:38 839次阅读

    辩论赛:智能家居生活变得简单还是麻烦

    如果两支队伍进行一场“智能家居生活变得简单还是麻烦”的辩论赛,会是什么情景? 如果这场辩论赛在本世纪之前开始,那么很大可能会无疾而终
    发表于 04-27 08:56 1527次阅读

    微软官宣正研究用户升级Windows10变得简便的方法

    事实上,微软正在通过一步步的努力,用户升级Windows 10变得容易,比如日前发布的Build 19536版本更新中,微软已经官宣正在研究简便的方法,来安装驱动程序和每月非安全质量更新。
    的头像 发表于 12-25 10:36 1893次阅读

    数字式万用表的测附件测量变得容易

    对数字式万用表的一个十分重要的要求是,它要能够使用各种各样的附件。很多附件可增加数字式万用表的测量范围和用途,同时测量变得容易。
    发表于 07-15 14:08 984次阅读

    如何MR变得落地、更易用

    科技 COO 程凌,通过回顾4年以来的创业开拓与研发创新历程,阐释如何 MR 变得落地、更易用,以及威廉希尔官方网站 升级所带来的多重价值。同时,还结合客户案例,分享 SaaS 服务的实际效能。
    的头像 发表于 03-11 09:19 1373次阅读

    关于Python 加速工具的选单

      这篇文章会提供一些 优化代码的工具 。会代码变得简洁,或者
    的头像 发表于 07-07 11:19 383次阅读

    分享一些Python代码加速工具

    这篇文章会提供一些优化代码的工具。会代码变得简洁,或者
    发表于 07-07 11:18 393次阅读

    视频处理器投影变得简单

    ,完美的把各组投影机的投影画面进行无缝边缘整合,而实现亮、更大、清的显示。 在投影威廉希尔官方网站 上的改进与创新成为信息科技威廉希尔官方网站 新的突破发向,而视频处理器的问世则投影变得更加简单,我们总结了
    的头像 发表于 08-24 16:37 825次阅读
    视频处理器<b class='flag-5'>让</b>投影<b class='flag-5'>变得</b><b class='flag-5'>更</b>简单

    Python开发者必备工具集

    这篇文章会提供一些优化代码的工具。会代码变得简洁,或者
    的头像 发表于 10-16 11:14 477次阅读

    “可靠”变得“更快安全”的数据传输协议:SCTP

    SCTP(Stream Control Transmission Protocol,流控传输协议)的出现,并不是万丈高楼平地起,而是站在TCP这个巨人肩膀上,数据传输从“可靠”变得“更快安全”。
    的头像 发表于 12-28 17:25 1432次阅读
    <b class='flag-5'>让</b>“可靠”<b class='flag-5'>变得</b>“更快<b class='flag-5'>更</b>安全”的数据传输协议:SCTP