成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

SpringBoot 如何統一后端返回格式?老鳥們都是這樣玩的!

開發 架構
今天我們來聊一聊在基于SpringBoot前后端分離開發模式下,如何友好的返回統一的標準格式以及如何優雅的處理全局異常。

[[411637]]

大家好,我是飄渺。

今天我們來聊一聊在基于SpringBoot前后端分離開發模式下,如何友好的返回統一的標準格式以及如何優雅的處理全局異常。

首先我們來看看為什么要返回統一的標準格式?

為什么要對SpringBoot返回統一的標準格式

在默認情況下,SpringBoot的返回格式常見的有三種:

第一種:返回 String

  1. @GetMapping("/hello"
  2. public String getStr(){ 
  3.   return "hello,javadaily"

此時調用接口獲取到的返回值是這樣:

  1. hello,javadaily 

第二種:返回自定義對象

  1. @GetMapping("/aniaml"
  2. public Aniaml getAniaml(){ 
  3.   Aniaml aniaml = new Aniaml(1,"pig"); 
  4.   return aniaml; 

此時調用接口獲取到的返回值是這樣:

  1.   "id": 1, 
  2.   "name""pig" 

第三種:接口異常

  1. @GetMapping("/error"
  2. public int error(){ 
  3.     int i = 9/0; 
  4.     return i; 

此時調用接口獲取到的返回值是這樣:

  1.   "timestamp""2021-07-08T08:05:15.423+00:00"
  2.   "status": 500, 
  3.   "error""Internal Server Error"
  4.   "path""/wrong" 

基于以上種種情況,如果你和前端開發人員聯調接口她們就會很懵逼,由于我們沒有給他一個統一的格式,前端人員不知道如何處理返回值。

還有甚者,有的同學比如小張喜歡對結果進行封裝,他使用了Result對象,小王也喜歡對結果進行包裝,但是他卻使用的是Response對象,當出現這種情況時我相信前端人員一定會抓狂的。

所以我們項目中是需要定義一個統一的標準返回格式的。

定義返回標準格式

一個標準的返回格式至少包含3部分:

  1. status 狀態值:由后端統一定義各種返回結果的狀態碼
  2. message 描述:本次接口調用的結果描述
  3. data 數據:本次返回的數據。
  1.   "status":"100"
  2.   "message":"操作成功"
  3.   "data":"hello,javadaily" 

當然也可以按需加入其他擴展值,比如我們就在返回對象中添加了接口調用時間

4.timestamp: 接口調用時間

定義返回對象

  1. @Data 
  2. public class ResultData<T> { 
  3.   /** 結果狀態 ,具體狀態碼參見ResultData.java*/ 
  4.   private int status; 
  5.   private String message; 
  6.   private T data; 
  7.   private long timestamp ; 
  8.  
  9.  
  10.   public ResultData (){ 
  11.     this.timestamp = System.currentTimeMillis(); 
  12.   } 
  13.  
  14.  
  15.   public static <T> ResultData<T> success(T data) { 
  16.     ResultData<T> resultData = new ResultData<>(); 
  17.     resultData.setStatus(ReturnCode.RC100.getCode()); 
  18.     resultData.setMessage(ReturnCode.RC100.getMessage()); 
  19.     resultData.setData(data); 
  20.     return resultData; 
  21.   } 
  22.  
  23.   public static <T> ResultData<T> fail(int code, String message) { 
  24.     ResultData<T> resultData = new ResultData<>(); 
  25.     resultData.setStatus(code); 
  26.     resultData.setMessage(message); 
  27.     return resultData; 
  28.   } 
  29.  

定義狀態碼

  1. public enum ReturnCode { 
  2.     /**操作成功**/ 
  3.     RC100(100,"操作成功"), 
  4.     /**操作失敗**/ 
  5.     RC999(999,"操作失敗"), 
  6.     /**服務限流**/ 
  7.     RC200(200,"服務開啟限流保護,請稍后再試!"), 
  8.     /**服務降級**/ 
  9.     RC201(201,"服務開啟降級保護,請稍后再試!"), 
  10.     /**熱點參數限流**/ 
  11.     RC202(202,"熱點參數限流,請稍后再試!"), 
  12.     /**系統規則不滿足**/ 
  13.     RC203(203,"系統規則不滿足要求,請稍后再試!"), 
  14.     /**授權規則不通過**/ 
  15.     RC204(204,"授權規則不通過,請稍后再試!"), 
  16.     /**access_denied**/ 
  17.     RC403(403,"無訪問權限,請聯系管理員授予權限"), 
  18.     /**access_denied**/ 
  19.     RC401(401,"匿名用戶訪問無權限資源時的異常"), 
  20.     /**服務異常**/ 
  21.     RC500(500,"系統異常,請稍后重試"), 
  22.  
  23.     INVALID_TOKEN(2001,"訪問令牌不合法"), 
  24.     ACCESS_DENIED(2003,"沒有權限訪問該資源"), 
  25.     CLIENT_AUTHENTICATION_FAILED(1001,"客戶端認證失敗"), 
  26.     USERNAME_OR_PASSWORD_ERROR(1002,"用戶名或密碼錯誤"), 
  27.     UNSUPPORTED_GRANT_TYPE(1003, "不支持的認證模式"); 
  28.  
  29.  
  30.  
  31.     /**自定義狀態碼**/ 
  32.     private final int code; 
  33.     /**自定義描述**/ 
  34.     private final String message; 
  35.  
  36.     ReturnCode(int code, String message){ 
  37.         this.code = code; 
  38.         this.message = message; 
  39.     } 
  40.  
  41.  
  42.     public int getCode() { 
  43.         return code; 
  44.     } 
  45.  
  46.     public String getMessage() { 
  47.         return message; 
  48.     } 

統一返回格式

  1. @GetMapping("/hello"
  2. public ResultData<String> getStr(){ 
  3.  return ResultData.success("hello,javadaily"); 

此時調用接口獲取到的返回值是這樣:

  1.   "status": 100, 
  2.   "message""hello,javadaily"
  3.   "data"null
  4.   "timestamp": 1625736481648, 
  5.   "httpStatus": 0 

這樣確實已經實現了我們想要的結果,我在很多項目中看到的都是這種寫法,在Controller層通過ResultData.success()對返回結果進行包裝后返回給前端。

看到這里我們不妨停下來想想,這樣做有什么弊端呢?

最大的弊端就是我們后面每寫一個接口都需要調用ResultData.success()這行代碼對結果進行包裝,重復勞動,浪費體力;

而且還很容易被其他老鳥給嘲笑。

所以呢我們需要對代碼進行優化,目標就是不要每個接口都手工制定ResultData返回值。

高級實現方式

要優化這段代碼很簡單,我們只需要借助SpringBoot提供的ResponseBodyAdvice即可。

ResponseBodyAdvice的作用:攔截Controller方法的返回值,統一處理返回值/響應體,一般用來統一返回格式,加解密,簽名等等。

先來看下ResponseBodyAdvice的源碼:

  1. public interface ResponseBodyAdvice<T> { 
  2.   /** 
  3.   * 是否支持advice功能 
  4.   * true 支持,false 不支持 
  5.   */ 
  6.     boolean supports(MethodParameter var1, Class<? extends HttpMessageConverter<?>> var2); 
  7.  
  8.    /** 
  9.   * 對返回的數據進行處理 
  10.   */ 
  11.     @Nullable 
  12.     T beforeBodyWrite(@Nullable T var1, MethodParameter var2, MediaType var3, Class<? extends HttpMessageConverter<?>> var4, ServerHttpRequest var5, ServerHttpResponse var6); 

我們只需要編寫一個具體實現類即可

  1. /** 
  2.  * @author jam 
  3.  * @date 2021/7/8 10:10 上午 
  4.  */ 
  5. @RestControllerAdvice 
  6. public class ResponseAdvice implements ResponseBodyAdvice<Object> { 
  7.     @Autowired 
  8.     private ObjectMapper objectMapper; 
  9.  
  10.     @Override 
  11.     public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) { 
  12.         return true
  13.     } 
  14.  
  15.     @SneakyThrows 
  16.     @Override 
  17.     public Object beforeBodyWrite(Object o, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) { 
  18.         if(o instanceof String){ 
  19.             return objectMapper.writeValueAsString(ResultData.success(o)); 
  20.         }         
  21.         return ResultData.success(o); 
  22.     } 

需要注意兩個地方:

  • @RestControllerAdvice注解

@RestControllerAdvice是@RestController注解的增強,可以實現三個方面的功能:

  1. 全局異常處理
  2. 全局數據綁定
  3. 全局數據預處理
  • String類型判斷
  1. if(o instanceof String){ 
  2.   return objectMapper.writeValueAsString(ResultData.success(o)); 
  3. }  

這段代碼一定要加,如果Controller直接返回String的話,SpringBoot是直接返回,故我們需要手動轉換成json。

經過上面的處理我們就再也不需要通過ResultData.success()來進行轉換了,直接返回原始數據格式,SpringBoot自動幫我們實現包裝類的封裝。

  1. @GetMapping("/hello"
  2. public String getStr(){ 
  3.     return "hello,javadaily"

此時我們調用接口返回的數據結果為:

  1. @GetMapping("/hello"
  2. public String getStr(){ 
  3.   return "hello,javadaily"

是不是感覺很完美,別急,還有個問題在等著你呢。

接口異常問題

此時有個問題,由于我們沒對Controller的異常進行處理,當我們調用的方法一旦出現異常,就會出現問題,比如下面這個接口

  1. @GetMapping("/wrong"
  2. public int error(){ 
  3.     int i = 9/0; 
  4.     return i; 

返回的結果為:

這顯然不是我們想要的結果,接口都報錯了還返回操作成功的響應碼,前端看了會打人的。

別急,接下來我們進入第二個議題,如何優雅的處理全局異常。

SpringBoot為什么需要全局異常處理器

不用手寫try...catch,由全局異常處理器統一捕獲

使用全局異常處理器最大的便利就是程序員在寫代碼時不再需要手寫try...catch了,前面我們講過,默認情況下SpringBoot出現異常時返回的結果是這樣:

  1.   "timestamp""2021-07-08T08:05:15.423+00:00"
  2.   "status": 500, 
  3.   "error""Internal Server Error"
  4.   "path""/wrong" 

這種數據格式返回給前端,前端是看不懂的,所以這時候我們一般通過try...catch來處理異常

  1. @GetMapping("/wrong"
  2. public int error(){ 
  3.     int i; 
  4.     try{ 
  5.         i = 9/0; 
  6.     }catch (Exception e){ 
  7.         log.error("error:{}",e); 
  8.         i = 0; 
  9.     } 
  10.     return i; 

我們追求的目標肯定是不需要再手動寫try...catch了,而是希望由全局異常處理器處理。

對于自定義異常,只能通過全局異常處理器來處理

  1. @GetMapping("error1"
  2. public void empty(){ 
  3.  throw  new RuntimeException("自定義異常"); 

當我們引入Validator參數校驗器的時候,參數校驗不通過會拋出異常,此時是無法用try...catch捕獲的,只能使用全局異常處理器。

SpringBoot集成參數校驗請參考這篇文章SpringBoot開發秘籍 - 集成參數校驗及高階技巧

如何實現全局異常處理器

  1. @Slf4j 
  2. @RestControllerAdvice 
  3. public class RestExceptionHandler { 
  4.     /** 
  5.      * 默認全局異常處理。 
  6.      * @param e the e 
  7.      * @return ResultData 
  8.      */ 
  9.     @ExceptionHandler(Exception.class) 
  10.     @ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR) 
  11.     public ResultData<String> exception(Exception e) { 
  12.         log.error("全局異常信息 ex={}", e.getMessage(), e); 
  13.         return ResultData.fail(ReturnCode.RC500.getCode(),e.getMessage()); 
  14.     } 
  15.  

有三個細節需要說明一下:

  1. @RestControllerAdvice,RestController的增強類,可用于實現全局異常處理器
  2. @ExceptionHandler,統一處理某一類異常,從而減少代碼重復率和復雜度,比如要獲取自定義異常可以@ExceptionHandler(BusinessException.class)
  3. @ResponseStatus指定客戶端收到的http狀態碼

體驗效果

這時候我們調用如下接口:

  1. @GetMapping("error1"
  2. public void empty(){ 
  3.     throw  new RuntimeException("自定義異常"); 

返回的結果如下:

  1.   "status": 500, 
  2.   "message""自定義異常"
  3.   "data"null
  4.   "timestamp": 1625795902556 

基本滿足我們的需求了。

但是當我們同時啟用統一標準格式封裝功能ResponseAdvice和RestExceptionHandler全局異常處理器時又出現了新的問題:

  1.   "status": 100, 
  2.   "message""操作成功"
  3.   "data": { 
  4.     "status": 500, 
  5.     "message""自定義異常"
  6.     "data"null
  7.     "timestamp": 1625796167986 
  8.   }, 
  9.   "timestamp": 1625796168008 

此時返回的結果是這樣,統一格式增強功能會給返回的異常結果再次封裝,所以接下來我們需要解決這個問題。

全局異常接入返回的標準格式

要讓全局異常接入標準格式很簡單,因為全局異常處理器已經幫我們封裝好了標準格式,我們只需要直接返回給客戶端即可。

  1. @SneakyThrows 
  2. @Override 
  3. public Object beforeBodyWrite(Object o, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) { 
  4.   if(o instanceof String){ 
  5.     return objectMapper.writeValueAsString(ResultData.success(o)); 
  6.   } 
  7.   if(o instanceof ResultData){ 
  8.     return o; 
  9.   } 
  10.   return ResultData.success(o); 

關鍵代碼:

  1. if(o instanceof ResultData){ 
  2.   return o; 

如果返回的結果是ResultData對象,直接返回即可。

這時候我們再調用上面的錯誤方法,返回的結果就符合我們的要求了。

  1.   "status": 500, 
  2.   "message""自定義異常"
  3.   "data"null
  4.   "timestamp": 1625796580778 

好了,今天的文章就到這里了,希望通過這篇文章你能掌握如何在你項目中友好實現統一標準格式到返回并且可以優雅的處理全局異常。

 

 

 

github地址:https://github.com/jianzh5/cloud-blog/

 

責任編輯:武曉燕 來源: JAVA日知錄
相關推薦

2023-11-23 08:25:31

String性能

2023-04-12 08:56:37

RocketMQSpring核心業務

2023-02-04 10:08:40

2015-07-15 13:47:13

上網高手

2022-08-31 08:19:04

接口returnCode代碼

2019-09-29 10:23:09

APIJava編程語言

2024-06-13 08:19:08

Controller接口參數

2020-11-16 13:38:31

PostMessage

2022-05-30 08:03:06

后端參數校驗異常處理

2021-10-11 19:34:03

全局格式項目

2021-07-28 06:10:47

拖拽設計器 transmat

2021-09-05 07:55:37

前端Emoji 表情

2024-08-06 09:51:21

SpringHTTPJSON

2015-04-16 09:48:12

APP測試

2024-08-02 08:38:20

Controller接口地址

2017-11-13 12:02:33

程序猿華為軟件

2019-05-30 14:58:56

Pythonxml文件

2023-11-28 14:32:04

2022-12-30 08:49:41

SpringBoot@Validated

2018-08-27 15:53:43

編程語言javaPython
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 一区二区三区电影在线观看 | 亚洲视频在线观看 | 亚洲一区在线播放 | 亚欧洲精品在线视频免费观看 | 国产亚洲欧美在线 | 国产精品久久精品 | 国产伊人久久久 | 国产二区av | 国产a区| 91久久精品 | 日韩一级免费电影 | 国产成人小视频 | 免费成人国产 | 黄篇网址| 国产一伦一伦一伦 | 天堂在线www | 亚洲自拍一区在线观看 | 国产草草视频 | 日韩不卡视频在线观看 | 成人精品国产一区二区4080 | 亚洲激情综合网 | 国产综合精品一区二区三区 | 久久伊人在 | 婷婷久久综合 | 久久久免费电影 | 欧美 日韩 在线播放 | 国产在线观看一区 | 一区二区免费看 | 一级片毛片 | 精品在线一区二区三区 | 国产视频在线观看一区二区三区 | 日韩一区二区三区精品 | 国产成人高清在线观看 | 一级a性色生活片久久毛片 一级特黄a大片 | 国产精品一卡 | 欧美日韩精品一区 | 91一区二区三区 | h视频免费在线观看 | 国产999在线观看 | h片在线看| 免费一区二区三区 |