激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - SpringBoot實戰之實現結果的優雅響應案例詳解

SpringBoot實戰之實現結果的優雅響應案例詳解

2021-12-16 12:09沉潛飛動 Java教程

這篇文章主要介紹了SpringBoot實戰之實現結果的優雅響應案例詳解,本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內容,需要的朋友可以參考下

今天說一下 Spring Boot 如何實現優雅的數據響應:統一的結果響應格式、簡單的數據封裝。

前提

無論系統規模大小,大部分 Spring Boot 項目是提供 Restful + json 接口,供前端或其他服務調用,格式統一規范,是程序猿彼此善待彼此的象征,也是減少聯調挨罵的基本保障。

通常響應結果中需要包含業務狀態碼、響應描述、響應時間戳、響應內容,比如:

{
"code": 200,
"desc": "查詢成功",
"timestamp": "2020-08-12 14:37:11",
"data": {
"uid": "1597242780874",
"name": "測試 1"
}
}

對于業務狀態碼分為兩個派系:一個是推薦使用 HTTP 響應碼作為接口業務返回;另一種是 HTTP 響應碼全部返回 200,在響應體中通過單獨的字段表示響應狀態。兩種方式各有優劣,個人推薦使用第二種,因為很多 Web 服務器對 HTTP 狀態碼有攔截處理功能,而且狀態碼數量有限,不夠靈活。比如返回 200 表示接口處理成功且正常響應,現在需要有一個狀態碼表示接口處理成功且正常響應,但是請求數據狀態不對,可以返回 2001 表示。

自定義響應體

定義一個數據響應體是返回統一響應格式的第一步,無論接口正常返回,還是發生異常,返回給調用方的結構格式都應該不變。給出一個示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
@ApiModel
@Data
public class Response<T> {
    @ApiModelProperty(value = "返回碼", example = "200")
    private Integer code;
    @ApiModelProperty(value = "返回碼描述", example = "ok")
    private String desc;
    @ApiModelProperty(value = "響應時間戳", example = "2020-08-12 14:37:11")
    private Date timestamp = new Date();
    @ApiModelProperty(value = "返回結果")
    private T data;
}

這樣,只要在 Controller 的方法返回Response就可以了,接口響應就一致了,但是這樣會形成很多格式固定的代碼模板,比如下面這種寫法:

?
1
2
3
4
5
6
7
8
@RequestMapping("hello1")
public Response<String> hello1() {
    final Response<String> response = new Response<>();
    response.setCode(200);
    response.setDesc("返回成功");
    response.setData("Hello, World!");
    return response;
}

調用接口響應結果為:

{
"code": 200,
"desc": "返回成功",
"timestamp": "2020-08-12 14:37:11",

     "data": "Hello, World!"

}

這種重復且沒有技術含量的代碼,怎么能配得上程序猿這種優(lan)雅(duo)的生物呢?最好能在返回響應結果的前提下,減去那些重復的代碼,比如:

?
1
2
3
4
@RequestMapping("hello2")
public String hello2() {
    return "Hello, World!";
}

這就需要借助 Spring 提供的ResponseBodyAdvice來實現了。

全局處理響應數據

先上代碼:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
/**
 * <br>created at 2020/8/12
 *
 * @author www.howardliu.cn
 * @since 1.0.0
 */
@RestControllerAdvice
public class ResultResponseAdvice implements ResponseBodyAdvice<Object> {
    @Override
    public boolean supports(final MethodParameter returnType, final Class<? extends HttpMessageConverter<?>> converterType) {
        return !returnType.getGenericParameterType().equals(Response.class);// 1
    }
 
    @Override
    public Object beforeBodyWrite(final Object body, final MethodParameter returnType, final MediaType selectedContentType,
                                  final Class<? extends HttpMessageConverter<?>> selectedConverterType,
                                  final ServerHttpRequest request, final ServerHttpResponse response) {
        if (body == null || body instanceof Response) {
            return body;
        }
        final Response<Object> result = new Response<>();
        result.setCode(200);
        result.setDesc("查詢成功");
        result.setData(body);
        if (returnType.getGenericParameterType().equals(String.class)) {// 2
            ObjectMapper objectMapper = new ObjectMapper();
            try {
                return objectMapper.writeValueAsString(result);
            } catch (JsonProcessingException e) {
                throw new RuntimeException("將 Response 對象序列化為 json 字符串時發生異常", e);
            }
        }
        return result;
    }
}
 
/**
 * <br>created at 2020/8/12
 *
 * @author www.howardliu.cn
 * @since 1.0.0
 */
@RestController
public class HelloWorldController {
    @RequestMapping("hello2")
    public String hello2() {
        return "Hello, World!";
    }
 
    @RequestMapping("user1")
    public User user1() {
        User u = new User();
        u.setUid(System.currentTimeMillis() + "");
        u.setName("測試1");
        return u;
    }
}

上面代碼是實現了 Spring ResponseBodyAdvice類的模板方式,按照 Spring 的要求實現就行。只有兩個需要特別注意的地方,也就是代碼中標注 1 和 2 的地方。

首先說 1 這一行,也就是supports方法,這個方法是校驗是否需要調用beforeBodyWrite方法的前置判斷,返回true則執行beforeBodyWrite方法,這里根據 Controller 方法返回類型來判斷是否需要執行beforeBodyWrite,也可以一律返回true,在后面判斷是否需要進行類型轉換。

然后重點說下 2 這一行,這行是坑,是大坑,如果對 Spring 結構不熟悉的,絕對會在這徘徊許久,不得妙法。

代碼 2 這一行是判斷Controller的方法是否返回的是String類型的結果,如果是,將返回的對象序列化之后返回。

這是因為SpringString類型的響應類型單獨處理了,使用StringHttpMessageConverter類進行數據轉換。在處理響應結果的時候,會在方法getContentLength中計算響應體大小,其父類方法定義是protected Long getContentLength(T t, @Nullable MediaType contentType),而StringHttpMessageConverter將方法重寫為protected Long getContentLength(String str, @Nullable MediaType contentType),第一個參數是響應對象,固定寫死是String類型,如果我們強制返回Response對象,就會報ClassCastException

當然,直接返回String的場景不多,這個坑可能會在某天特殊接口中突然出現。

補充說明

上面只是展示了ResponseBodyAdvice類最簡單的應用,我們還可以實現更多的擴展使用。比如:

  1. 返回請求ID:這個需要與與RequestBodyAdvice聯動,獲取到請求ID后,在響應是放在響應體中;
  2. 結果數據加密:通過ResponseBodyAdvice實現響應數據加密,不會侵入業務代碼,而且可以通過注解方式靈活處理接口的加密等級;
  3. 有選擇的包裝響應體:比如定義注解IgnoreResponseWrap,在不需要包裝響應體的接口上定義,然后在supports方法上判斷方法的注解即可,比如:
?
1
2
3
4
5
@Override
public boolean supports(final MethodParameter returnType, final Class<? extends HttpMessageConverter<?>> converterType) {
    final IgnoreResponseWrap[] declaredAnnotationsByType = returnType.getExecutable().getDeclaredAnnotationsByType(IgnoreResponseWrap.class);
    return !(declaredAnnotationsByType.length > 0 || returnType.getGenericParameterType().equals(Response.class));
}

很多其他玩法就不一一列舉了。

總結

上面說了正常響應的數據,只做到了一點優雅,想要完整,還需要考慮接口異常情況,總不能來個大大的try/catch/finally包住業務邏輯吧,那也太丑了。后面會再來一篇,重點說說接口如何在出現異常時,也能返回統一的結果響應。

本文只是拋出一塊磚,玉還得自己去找。

推薦閱讀

到此這篇關于SpringBoot實戰之實現結果的優雅響應案例詳解的文章就介紹到這了,更多相關SpringBoot實戰之實現結果的優雅響應內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!

原文鏈接:https://www.howardliu.cn/springboot-action-gracefully-response/

延伸 · 閱讀

精彩推薦
  • Java教程小米推送Java代碼

    小米推送Java代碼

    今天小編就為大家分享一篇關于小米推送Java代碼,小編覺得內容挺不錯的,現在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧...

    富貴穩中求8032021-07-12
  • Java教程Java BufferWriter寫文件寫不進去或缺失數據的解決

    Java BufferWriter寫文件寫不進去或缺失數據的解決

    這篇文章主要介紹了Java BufferWriter寫文件寫不進去或缺失數據的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望...

    spcoder14552021-10-18
  • Java教程20個非常實用的Java程序代碼片段

    20個非常實用的Java程序代碼片段

    這篇文章主要為大家分享了20個非常實用的Java程序片段,對java開發項目有所幫助,感興趣的小伙伴們可以參考一下 ...

    lijiao5352020-04-06
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    這篇文章主要介紹了Java使用SAX解析xml的示例,幫助大家更好的理解和學習使用Java,感興趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程升級IDEA后Lombok不能使用的解決方法

    升級IDEA后Lombok不能使用的解決方法

    最近看到提示IDEA提示升級,尋思已經有好久沒有升過級了。升級完畢重啟之后,突然發現好多錯誤,本文就來介紹一下如何解決,感興趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程xml與Java對象的轉換詳解

    xml與Java對象的轉換詳解

    這篇文章主要介紹了xml與Java對象的轉換詳解的相關資料,需要的朋友可以參考下...

    Java教程網2942020-09-17
  • Java教程Java8中Stream使用的一個注意事項

    Java8中Stream使用的一個注意事項

    最近在工作中發現了對于集合操作轉換的神器,java8新特性 stream,但在使用中遇到了一個非常重要的注意點,所以這篇文章主要給大家介紹了關于Java8中S...

    阿杜7482021-02-04
  • Java教程Java實現搶紅包功能

    Java實現搶紅包功能

    這篇文章主要為大家詳細介紹了Java實現搶紅包功能,采用多線程模擬多人同時搶紅包,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙...

    littleschemer13532021-05-16
主站蜘蛛池模板: 蜜桃av鲁一鲁一鲁一鲁 | 精精国产xxxx视频在线野外 | 久久色伦理资源站 | 色99久久| 国产1区2| 国产成人在线观看免费 | 久久无| 一区免费| 在线看91 | 九九热精品在线视频 | 欧美aaaaaaaa| 国产一区精品视频 | 一级免费在线视频 | 爱逼爱操综合网 | 在线a毛片| 国产69精品久久久久久野外 | 欧美日韩在线影院 | www.69色| 天天夜夜操操 | 国产精品中文在线 | 女人叉开腿让男人桶 | 国产午夜精品一区 | 91av大片 | 国产成人强伦免费视频网站 | 亚洲国产精品久久久久久久 | 亚洲成人福利网站 | 国产精品jk白丝蜜臀av软件 | 天天色狠狠干 | 一区二区三区欧洲 | 亚洲成人精品久久久 | 国产福利不卡一区二区三区 | 九九精品影院 | 欧美精品v国产精品v日韩精品 | 欧美爱爱视频网站 | 中文字幕偷拍 | 日本免费一区二区三区四区 | 成人午夜天堂 | 国产一区二区观看 | 日韩伦理电影免费观看 | 免费观看又色又爽又黄的崩锅 | 日本不卡一二三区 |