目錄
一、SpringBoot不使用統一回傳格式
1.1 使用字串回傳
1.2 使用實體類別回傳
1.3 異常情況下回傳
二、基礎玩法
2.1 參數說明
資料回傳格式
@ResponseBody
Result
3.4 统一返回结果处理类最终版
首頁 Java java教程 SpringBoot統一介面回傳及全域異常如何處理

SpringBoot統一介面回傳及全域異常如何處理

May 12, 2023 pm 04:01 PM
springboot

一、SpringBoot不使用統一回傳格式

預設情況下,SpringBoot會有以下三種回傳情況。

1.1 使用字串回傳

@GetMapping("/getUserName")
public String getUserName(){
    return "HuaGe";
}
登入後複製

呼叫介面傳回結果:

HuaGe
登入後複製

1.2 使用實體類別回傳

@GetMapping("/getUserName")
public User getUserName(){
    return new User("HuaGe",18,"男");
}
登入後複製

呼叫介面傳回結果:

{
  "name": "HuaGe",
  "age": "18",
  "性别": "男", 
}
登入後複製

1.3 異常情況下回傳

@GetMapping("/getUserName")
public static String getUserName(){
    HashMap hashMap = Maps.newHashMap();
    return hashMap.get(0).toString();
}
登入後複製

模擬一個空指標異常,在不做任何異常處理的情況下,可以看下SpringBoot的預設回傳結果:

{
    "timestamp": "2021-08-09T06:56:41.524+00:00",
    "status": 500,
    "error": "Internal Server Error",
    "path": "/sysUser/getUserName"
}
登入後複製

對於上面這幾種情況,如果整個專案沒有定義統一的返回格式,五個後台開發人員定義五種返回格式,這樣不僅代碼臃腫,前後端對接效率低,而且還會有一些意向不到的情況發生,例如前端直接顯示異常詳情等,這給用戶的體驗是非常差的。

二、基礎玩法

專案中最常見到的是封裝一個工具類,類別中定義需要返回的字段信息,把需要返回前端的接口信息,通過該類進行封裝,這樣就可以解決返回格式不統一的現象了。

2.1 參數說明

  • code: 狀態碼,後台可以維護一套統一的狀態碼;

  • ##message: 描述訊息,介面呼叫成功/失敗的提示訊息;

  • data: 傳回資料。

2.2 流程說明

  • #新建Result類別

    ##
    public class Result<T> {
        
        private int code;
        
        private String message;
        
        private T data;
    
        public Result() {}
        public Result(int code, String message) {
            this.code = code;
            this.message = message;
        }
        
        /**
         * 成功
         */
        public static <T> Result<T> success(T data) {
            Result<T> result = new Result<T>();
            result.setCode(ResultMsgEnum.SUCCESS.getCode());
            result.setMessage(ResultMsgEnum.SUCCESS.getMessage());
            result.setData(data);
            return result;
        }
    
        /**
         * 失败
         */
        public static <T> Result<T> error(int code, String message) {
            return new Result(code, message);
        }
    }
    登入後複製

定義回傳狀態碼

public enum ResultMsgEnum {
    SUCCESS(0, "成功"),
    FAIL(-1, "失败"),
    AUTH_ERROR(502, "授权失败!"),
    SERVER_BUSY(503, "服务器正忙,请稍后再试!"),
    DATABASE_OPERATION_FAILED(504, "数据库操作失败");
    private int code;
    private String message;

    ResultMsgEnum(int code, String message) {
        this.code = code;
        this.message = message;
    }
    public int getCode() {
        return this.code;
    }
    
    public String getMessage() {
        return this.message;
    }
}
登入後複製

使用方式

上面兩個步驟定義了

資料回傳格式

狀態碼

,接下來就要看下在介面中如何使用了。

@GetMapping("/getUserName")
public Result getUserName(){
    return Result.success("huage");
}
登入後複製

呼叫結果如下,可以看到是我們在Result中定義的參數類型。
    {
        "code": 0,
        "message": "成功",
        "data": "huage"
    }
    登入後複製
  • 這樣寫雖然能夠滿足日常需求,而且我相信很多小伙伴也是這麼用的,但是如果我們有大量的接口,然後在每一個接口中都使用

    Result.success來包裝回訊息,會新增很多重複程式碼,顯得不夠優雅,甚至都不好意思拿出去顯擺。肯定會有一種方式能夠再一次提高程式碼逼格,實現最優解。 三、進階用法

    基本用法學會後,接下來看點究極版本,主要用到如下兩個知識點,用法簡單,無論是拿出來教學妹,還是指點小姐姐,都是必備技能。
  • 3.1 類別介紹

  • ResponseBodyAdvice:
此介面是SpringMVC 4.1提供的,它允許在執行

@ResponseBody

後自訂返回數據,用來封裝統一資料格式傳回;
  • @RestControllerAdvice:

    此註解是對Controller進行增強的,可以全域擷取拋出的異常。
  • 3.2 用法說明ResponseAdvice

    類別;
  • 實現
  • ResponseBodyAdvice
接口,實作

supportsbeforeBodyWrite方法;

該類別用於統一封裝controller中介面的傳回結果。

@RestControllerAdvice
public class ResponseAdvice implements ResponseBodyAdvice<Object> {
    @Autowired
    private ObjectMapper objectMapper;

    /**
     * 是否开启功能 true:是 
     */
    @Override
    public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
        return true;
    }

    /**
     * 处理返回结果
     */
    @Override
    public Object beforeBodyWrite(Object o, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
        //处理字符串类型数据
        if(o instanceof String){
            try {
                return objectMapper.writeValueAsString(Result.success(o));
            } catch (JsonProcessingException e) {
                e.printStackTrace();
            }
        }
        return Result.success(o);
    }
}
登入後複製

我們可以透過

getUserName

介面測試一下,會發現和直接使用

Result

傳回的結果是一致的。

不過,細心的小夥伴們肯定注意到了,在

ResponseAdvice我們全部使用了Result.success(o)

來處理結果,對於error類型的結果未做處理。我們來看下,發生異常情況時,回傳結果是什麼樣子呢?繼續使用上面HashMap空指標異常的程式碼,測試結果如下:

{
    "code": 0,
    "message": "成功",
    "data": {
        "timestamp": "2021-08-09T09:33:26.805+00:00",
        "status": 405,
        "error": "Method Not Allowed",
        "path": "/sysUser/getUserName"
    }
}
登入後複製
雖然格式上沒有毛病,但在code、data欄位的特定資料上是不友善或不正確的。不處理好這些事情,會嚴重影響自己在前端妹妹心中的高大形象的,這是絕對不能容忍的。

3.3 全域例外處理器

以前我們遇到例外時,第一時間想到的應該是try..catch..finnal吧,不過這種方式會導致大量程式碼重複,維護困難,邏輯臃腫等問題,這不是我們想要的結果。 今天我們要用的全域異常處理方式,用起來是比較簡單的。首先新增一個類,增加@RestControllerAdvice

註解,該註解的作用花哥上面已經介紹過,就不再嘮叨了。 ###
@RestControllerAdvice
public class CustomerExceptionHandler {
    
}
登入後複製
###如果我們有想要攔截的例外類型,就新增一個方法,使用###@ExceptionHandler###註解修飾,註解參數為目標例外類型。 ######例如:controller中介面發生Exception異常時,就會進入到###Execption###方法中進行捕獲,將雜亂的異常訊息,轉換成指定格式後交給###ResponseAdvice# ##方法進行統一格式封裝並回傳給前端小夥伴。 ###
@RestControllerAdvice
@Slf4j
public class CustomerExceptionHandler {

    @ExceptionHandler(AuthException.class)
    public String ErrorHandler(AuthorizationException e) {
        log.error("没有通过权限验证!", e);
        return "没有通过权限验证!";
    }

    @ExceptionHandler(Exception.class)
    public Result Execption(Exception e) {
        log.error("未知异常!", e);
        return Result.error(ResultMsgEnum.SERVER_BUSY.getCode(),ResultMsgEnum.SERVER_BUSY.getMessage());
    }
}
登入後複製

再次调用接口getUserName查看返回结果,会发现还是有一些问题,因为我们在CustomerExceptionHandler中已经将接口返回结果封装成Result类型,而代码执行到统一结果返回类ResponseAdvice时,又会结果再次封装,就出现了如下问题。

{
    "code": 0,
    "message": "成功",
    "data": {
        "code": 503,
        "message": "服务器正忙,请稍后再试!",
        "data": null
    }
}
登入後複製

3.4 统一返回结果处理类最终版

解决上述问题非常简单,只要在beforeBodyWrite中增加一条判断即可。

@RestControllerAdvice
public class ResponseAdvice implements ResponseBodyAdvice<Object> {
    @Autowired
    private ObjectMapper objectMapper;

    /**
     * 是否开启功能 true:开启
     */
    @Override
    public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> aClass) {
        return true;
    }

    /**
     * 处理返回结果
     */
    @Override
    public Object beforeBodyWrite(Object o, MethodParameter methodParameter, MediaType mediaType, Class<? extends HttpMessageConverter<?>> aClass, ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
        //处理字符串类型数据
        if(o instanceof String){
            try {
                return objectMapper.writeValueAsString(Result.success(o));
            } catch (JsonProcessingException e) {
                e.printStackTrace();
            }
        }
        //返回类型是否已经封装
        if(o instanceof Result){
            return o;
        }
        return Result.success(o);
    }
}
登入後複製

以上是SpringBoot統一介面回傳及全域異常如何處理的詳細內容。更多資訊請關注PHP中文網其他相關文章!

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1665
14
CakePHP 教程
1424
52
Laravel 教程
1321
25
PHP教程
1269
29
C# 教程
1249
24
Springboot怎麼整合Jasypt實現設定檔加密 Springboot怎麼整合Jasypt實現設定檔加密 Jun 01, 2023 am 08:55 AM

Jasypt介紹Jasypt是一個java庫,它允許開發員以最少的努力為他/她的專案添加基本的加密功能,並且不需要對加密工作原理有深入的了解用於單向和雙向加密的高安全性、基於標準的加密技術。加密密碼,文本,數字,二進位檔案...適合整合到基於Spring的應用程式中,開放API,用於任何JCE提供者...添加如下依賴:com.github.ulisesbocchiojasypt-spring-boot-starter2. 1.1Jasypt好處保護我們的系統安全,即使程式碼洩露,也可以保證資料來源的

怎麼在SpringBoot中使用Redis實現分散式鎖 怎麼在SpringBoot中使用Redis實現分散式鎖 Jun 03, 2023 am 08:16 AM

一、Redis實現分散式鎖原理為什麼需要分散式鎖在聊分散式鎖之前,有必要先解釋一下,為什麼需要分散式鎖。與分散式鎖相對就的是單機鎖,我們在寫多執行緒程式時,避免同時操作一個共享變數產生資料問題,通常會使用一把鎖來互斥以保證共享變數的正確性,其使用範圍是在同一個進程中。如果換做是多個進程,需要同時操作一個共享資源,如何互斥?現在的業務應用通常是微服務架構,這也意味著一個應用會部署多個進程,多個進程如果需要修改MySQL中的同一行記錄,為了避免操作亂序導致髒數據,此時就需要引入分佈式鎖了。想要實現分

SpringBoot怎麼整合Redisson實現延遲隊列 SpringBoot怎麼整合Redisson實現延遲隊列 May 30, 2023 pm 02:40 PM

使用場景1、下單成功,30分鐘未支付。支付超時,自動取消訂單2、訂單簽收,簽收後7天未進行評估。訂單超時未評價,系統預設好評3、下單成功,商家5分鐘未接單,訂單取消4、配送超時,推播簡訊提醒…對於延時比較長的場景、即時性不高的場景,我們可以採用任務調度的方式定時輪詢處理。如:xxl-job今天我們採

springboot讀取檔案打成jar包後存取不到怎麼解決 springboot讀取檔案打成jar包後存取不到怎麼解決 Jun 03, 2023 pm 04:38 PM

springboot讀取文件,打成jar包後訪問不到最新開發出現一種情況,springboot打成jar包後讀取不到文件,原因是打包之後,文件的虛擬路徑是無效的,只能通過流去讀取。文件在resources下publicvoidtest(){Listnames=newArrayList();InputStreamReaderread=null;try{ClassPathResourceresource=newClassPathResource("name.txt");Input

Springboot+Mybatis-plus不使用SQL語句進行多表新增怎麼實現 Springboot+Mybatis-plus不使用SQL語句進行多表新增怎麼實現 Jun 02, 2023 am 11:07 AM

在Springboot+Mybatis-plus不使用SQL語句進行多表添加操作我所遇到的問題準備工作在測試環境下模擬思維分解一下:創建出一個帶有參數的BrandDTO對像模擬對後台傳遞參數我所遇到的問題我們都知道,在我們使用Mybatis-plus中進行多表操作是極其困難的,如果你不使用Mybatis-plus-join這一類的工具,你只能去配置對應的Mapper.xml文件,配置又臭又長的ResultMap,然後再寫對應的sql語句,這種方法雖然看上去很麻煩,但具有很高的靈活性,可以讓我們

SpringBoot怎麼自訂Redis實作快取序列化 SpringBoot怎麼自訂Redis實作快取序列化 Jun 03, 2023 am 11:32 AM

1.自訂RedisTemplate1.1、RedisAPI預設序列化機制基於API的Redis快取實作是使用RedisTemplate範本進行資料快取操作的,這裡開啟RedisTemplate類,查看該類別的源碼資訊publicclassRedisTemplateextendsRedisAccessorimplementsRedisOperations,BeanClassLoaderAware{//聲明了value的各種序列化方式,初始值為空@NullableprivateRedisSe

SpringBoot與SpringMVC的比較及差別分析 SpringBoot與SpringMVC的比較及差別分析 Dec 29, 2023 am 11:02 AM

SpringBoot和SpringMVC都是Java開發中常用的框架,但它們之間有一些明顯的差異。本文將探究這兩個框架的特點和用途,並對它們的差異進行比較。首先,我們來了解一下SpringBoot。 SpringBoot是由Pivotal團隊開發的,它旨在簡化基於Spring框架的應用程式的建立和部署。它提供了一種快速、輕量級的方式來建立獨立的、可執行

springboot怎麼取得application.yml裡值 springboot怎麼取得application.yml裡值 Jun 03, 2023 pm 06:43 PM

在專案中,很多時候需要用到一些配置信息,這些信息在測試環境和生產環境下可能會有不同的配置,後面根據實際業務情況有可能還需要再做修改。我們不能將這些設定在程式碼中寫死,最好是寫到設定檔中,例如可以把這些資訊寫到application.yml檔案中。那麼,怎麼在程式碼裡取得或使用這個位址呢?有2個方法。方法一:我們可以透過@Value註解的${key}即可取得設定檔(application.yml)中和key對應的value值,這個方法適用於微服務比較少的情形方法二:在實際專案中,遇到業務繁瑣,邏

See all articles