.net core 數據統一響應分享

来源:https://www.cnblogs.com/WNpursue/archive/2020/04/13/12693466.html
-Advertisement-
Play Games

前言 代碼胡亂寫,維護火葬場! 在平時工作中遇到前同事寫介面這樣返回值 當介面返回1時,不去看他的代碼就永遠猜不到這個1代表的是返回成功還是返回值 稍微好點的 維護和改bug簡直讓人瘋狂,導致大部分時間浪費在“體會”別人返回值的邏輯中 天天加班與救bug於水火之中 合理分配前後端返回值很重要! 一般 ...


前言

代碼胡亂寫,維護火葬場!

在平時工作中遇到前同事寫介面這樣返回值


當介面返回1時,不去看他的代碼就永遠猜不到這個1代表的是返回成功還是返回值

稍微好點的

維護和改bug簡直讓人瘋狂,導致大部分時間浪費在“體會”別人返回值的邏輯中

天天加班與救bug於水火之中

合理分配前後端返回值很重要!

一般我們常用統一的格式返回json,無論後臺是運行正常還是發生異常,響應給前端的數據格式是不變的!

    public class Result<T> where T : class
    {
        public ResultCode code { get; set; }

        public string msg { get; set; }

        public T data { get; set; }
    }

包裝一下,通常使用構造函數

    public class Result<T> where T : class
    {
        public ResultCode code { get; set; }

        public string msg { get; set; }

        public T data { get; set; }

        public Result(T data)
        {
            code = ResultCode.Ok;
            msg = ResultCode.Ok.GetDescription();
            this.data = data;
        }

        public Result(ResultCode code, string msg, T data)
        {
            this.code = code;
            this.msg = msg ?? code.GetDescription();
            this.data = data;
        }
    }

這麼使用new Result<Store>(code, msg, data)使用起來還是不夠簡化
繼續封裝

    public class Result<T> where T : class
    {
        public ResultCode code { get; set; }

        public string msg { get; set; }

        public T data { get; set; }

        public Result(T data)
        {
            code = ResultCode.Ok;
            msg = ResultCode.Ok.GetDescription();
            this.data = data;
        }

        public Result(ResultCode code, string msg, T data)
        {
            this.code = code;
            this.msg = msg ?? code.GetDescription();
            this.data = data;
        }

        public static Result<T> FromCode(ResultCode code, string msg = null, T data = default)
        {
            if (data == null)
                data = (T) new Object();
            return new Result<T>(code, msg, data);
        }

        public static Result<T> Ok(T data)
        {
            return new Result<T>(data);
        }

        public static Result<T> FromError(ResultCode code = ResultCode.Fail, string msg = null, T data = default)
        {
            return new Result<T>(code, msg, data);
        }
    }
        [HttpGet]
        public Result<Store> Get()
        {
            return Result<Store>.Ok(new Store());
        }

這樣就稍微好點了

全局處理響應數據

雖然控制器返回值可以統一了,但是異常處理並沒有統一,這時候就需要建個中間件/異常過濾器進行處理異常

    app.UseMiddleware<GlobalExceptionHandlerMiddleware>();
    public class GlobalExceptionHandlerMiddleware
    {
        private readonly RequestDelegate _next;

        public GlobalExceptionHandlerMiddleware(RequestDelegate next)
        {
            _next = next;
        }

        public async Task Invoke(HttpContext context)
        {
            try
            {
                await _next(context);
            }
            catch (System.Exception ex)
            {
                context.Response.StatusCode = 500;
                context.Response.ContentType = "text/json;charset=utf-8;";

                if (ex is ResultException == false)
                {
                    var logger = context.RequestServices.GetRequiredService<ILoggerFactory>().CreateLogger<GlobalExceptionHandlerMiddleware>();
                    logger.LogError(1, ex, ex.Message);
                }

                var json = Result<object>.FromCode(ResultCode.Fail, ex.Message);
                var error = JsonSerializer.Serialize(json);

                await context.Response.WriteAsync(error);
            }
        }
    }

有些異常是不需要記錄的,這時候可以自定義ResultException進行判斷

    public class ResultException : Exception
    {
        public ResultException(string message) : base(message)
        {
        }

        public ResultException(string message, Exception e) : base(message, e)
        {

        }
    }

                if (ex is ResultException == false)
                {
                    var logger = context.RequestServices.GetRequiredService<ILoggerFactory>().CreateLogger<GlobalExceptionHandlerMiddleware>();
                    logger.LogError(1, ex, ex.Message);
                }

參數校驗處理

一個介面一般對參數(請求數據)都會進行安全校驗,按照之前的格式返回驗證異常。

    public class ModelActionFilter : ActionFilterAttribute, IActionFilter
    {
        public override void OnActionExecuting(ActionExecutingContext context)
        {
            if (!context.ModelState.IsValid)
            {
                var errorResults = new List<ErrorResultDto>();
                foreach (var item in context.ModelState)
                {
                    var result = new ErrorResultDto
                    {
                        Field = item.Key,
                        Msg = "",
                    };
                    foreach (var error in item.Value.Errors)
                    {
                        if (!string.IsNullOrEmpty(result.Msg))
                        {
                            result.Msg += '|';
                        }
                        result.Msg += error.ErrorMessage;
                    }
                    errorResults.Add(result);
                }
                context.Result = new JsonResult(Result<List<ErrorResultDto>>.FromCode(ResultCode.InvalidData, data: errorResults));
            }
        }
    }

    public class ErrorResultDto
    {
        /// <summary>
        /// 參數領域
        /// </summary>
        public string Field { get; set; }

        /// <summary>
        /// 錯誤信息
        /// </summary>
        public string Msg { get; set; }
    }

Startup.cs

            services.AddControllers(options =>
            {
                options.Filters.Add(new ModelActionFilter());
                options.MaxModelValidationErrors = 50;
                options.ModelBindingMessageProvider.SetValueMustNotBeNullAccessor(
                    _ => "該欄位不可為空。");
            })

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 函數式介面#第一章 函數式介面 ...
  • 前言 文的文字及圖片來源於網路,僅供學習、交流使用,不具有任何商業用途,版權歸原作者所有,如有問題請及時聯繫我們以作處理。 作者:kwsy PS:如有需要Python學習資料的小伙伴可以加點擊下方鏈接自行獲取http://t.cn/A6Zvjdun 1. python 封包 將多個值賦值給一個變數時 ...
  • MD5加密很常用,比如資料庫中密碼等敏感欄位需要加密存儲,核對密碼時先以同樣的方式對用戶輸入的密碼進行加密,再與資料庫中存儲的密碼比較。 MD5加密有很多種實現方式,此處介紹2種。 1、使用JDK自帶MessageDigest public class MD5Util { public static ...
  • 估計很多人在網上看到各種各樣的DeepClone實現, 例如: 1. 通過BinaryFormatter進行二進位序列化 這玩意兒序列化出來的東西還帶namespace類型, 尺寸非常大, 調試一下就知道極其不靠譜 有些人又開始動歪腦筋了, 說我搞一個JSON序列化, 或者BSON序列化可不可以 2 ...
  • 上一篇文章用python實現了計算文本相似度計算的過程,這次用C 做個demo。 不得不說用python是真的方便,不懂計算過程也能實現結果。C 也有類似NumPy的庫: "NumSharp" 。經過測試還是有區別的,有些功能沒有(也可能是因為我沒看文檔)。最後還是自己研究計算過程去寫。 用C 寫E ...
  • 標題可能有點不好理解,我解釋一下: 術語指的是常用辭彙,比如用戶(User)、名稱(Name)、郵箱(Email)等。 帶修飾符指的是UI上需要呈現`用戶:`、`名稱:`、`郵箱:`等這些情況。 由於之前已經定義過了用戶此類辭彙的多語言,如果還需要再對它們帶冒號的版本定義一次多語言,這個過程會非常冗 ...
  • WPF的UI多語言切換核心代碼: CultureInfo.CurrentUICulture = CultureInfo.CreateSpecificCulture("語種"); 關於上面的語種如何傳進去,什麼時候調用上面的方法是跟你具體業務有關的。 我的做法: 1. 什麼時候調用 因為用戶的語種基本 ...
  • 關於多語言,其實有很多經驗可以分享。本篇文章先說說最基礎的。 下圖:利用自帶的resx文件格式記錄各種通用辭彙的多語言(簡體中文、繁體中文、英文) 下圖是英文的示例(註意:Access Modifier必須是`Public`,否則無法在xaml文件中引用) 使用方法: 1. 聲明命名空間 xmlns ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...