.net core 自定义规范响应的中间件

在本文中,我们将介绍如何使用 .NET Core 中的中间件来自定义规范响应,以便在 API 调用时返回统一的格式和错误信息。中间件是一种可以在请求和响应管道中执行逻辑的软件组件,它可以对请求或响应进行修改、拦截或处理。我们将使用一个简单的示例来演示如何创建和使用自定义规范响应的中间件。

首先,我们需要创建一个类来表示规范响应的格式,这个类可以包含以下属性:

  • Code:响应的状态码,例如 200 表示成功,400 表示客户端错误,500 表示服务器错误等。
  • Message:响应的消息,例如 "OK" 表示成功,"Bad Request" 表示客户端错误,"Internal Server Error" 表示服务器错误等。
  • Data:响应的数据,可以是任意类型的对象,例如用户信息、产品列表、订单详情等。

这个类的代码如下:

public class ApiResponse
{
    public bool Success { get; set; }
    public string Message { get; set; }
    public object Data { get; set; }

    public ApiResponse(bool success, string message, object data)
    {
        Success = success;
        Message = message;
        Data = data;
    }

    public ApiResponse(bool success, string message)
        : this(success, message, null)
    {
    }

    public ApiResponse(bool success)
        : this(success, null, null)
    {
    }
}

中间件

接下来,我们需要创建一个中间件类来实现自定义规范响应的逻辑,这个类需要有以下特点:

  • 接收一个 RequestDelegate 类型的参数,表示下一个中间件或终端处理程序。
  • 实现一个 InvokeAsync 方法,接收一个 HttpContext 类型的参数,表示当前请求的上下文。
  • InvokeAsync 方法中,使用 await next(context) 来调用下一个中间件或终端处理程序,并获取其返回的响应。
  • InvokeAsync 方法中,根据响应的状态码和内容来构造一个 ApiResponse对象,并将其序列化为 JSON 格式。
  • InvokeAsync 方法中,修改响应的内容类型为 application/json,并将 JSON 格式的 ApiResponse写入到响应体中。
  • GetStatusCodeMessage()根据响应状态给出信息
  • GetResponseData()获取其返回的响应

CustomResponseMiddleware

public class CustomResponseMiddleware
{
    private readonly RequestDelegate _next;

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

    public async Task InvokeAsync(HttpContext context)
    {
        var originalBodyStream = context.Response.Body;

        using (var responseBody = new MemoryStream())
        {
            context.Response.Body = responseBody;

            await _next(context);

            if (context.Response.StatusCode >= 400 && context.Response.StatusCode <= 599)
            {
                context.Response.ContentType = "application/json";

                var response = new ApiResponse
                {
                    Success = false,
                    Message = GetStatusCodeMessage(context.Response.StatusCode),
                    Data = await GetResponseData(context.Response)
                };

                var jsonResponse = JsonConvert.SerializeObject(response);
                await context.Response.WriteAsync(jsonResponse, Encoding.UTF8);
            }
            else
            {
                context.Response.ContentType = "application/json";

                var response = new ApiResponse
                {
                    Success = true,
                    Message = GetStatusCodeMessage(context.Response.StatusCode),
                    Data = await GetResponseData(context.Response)
                };

                var jsonResponse = JsonConvert.SerializeObject(response);
                await context.Response.WriteAsync(jsonResponse, Encoding.UTF8);
            }

            await responseBody.CopyToAsync(originalBodyStream);
        }
    }
}

GetStatusCodeMessage()

 private static string GetStatusCodeMessage(int statusCode)
    {
        switch (statusCode)
        {
            case 200:
                return "OK";
            case 201:
                return "Created";
            case 204:
                return "No Content";
            case 400:
                return "Bad Request";
            case 401:
                return "Unauthorized";
            case 403:
                return "Forbidden";
            case 404:
                return "Not Found";
            case 500:
                return "Internal Server Error";
            default:
                return "Unknown Status Code";
        }
    }

GetResponseData()

private async Task<object> GetResponseData(HttpResponse response)
    {
        var body = await new StreamReader(response.Body).ReadToEndAsync();
        response.Body.Seek(0, SeekOrigin.Begin);

        try
        {
            return JsonConvert.DeserializeObject(body);
        }
        catch (JsonReaderException)
        {
            return new { Message = body };
        }
    }

在上面的示例中,我们创建了一个名为 CustomResponseMiddleware 的中间件。该中间件拦截每个响应,并根据需要修改响应格式。具体来说,如果响应的状态码为 4xx 或5xx,则中间件将返回一个包含错误消息和数据的 ApiResponse 对象;否则,中间件将返回一个包含成功消息和数据的 ApiResponse 对象。

常用类

定义常用的类可以帮助我们标准化 ASP.NET Core 应用程序中的响应格式,提高代码重用性,并使前端更加轻松地处理所有响应。

除了 ApiResponse 类之外,还可以定义其他常用类,如 ApiError 类、ApiResponse 泛型类等,以满足不同的需求。例如,ApiError 类可以用于标准化应用程序中的错误响应格式,ApiResponse 泛型类可以用于在响应中包含更具体的数据类型。

下面是 ApiError 类的示例代码:

public class ApiError
{
    public int StatusCode { get; set; }
    public string Message { get; set; }

    public override string ToString()
    {
        return JsonConvert.SerializeObject(this);
    }
}

ApiError 类包含两个属性:StatusCodeMessageStatusCode 属性指示错误的状态码,Message 属性包含有关错误的消息。

使用 ApiError 类可以帮助我们标准化应用程序中的错误响应格式。例如,在某些情况下,我们可能需要返回一个包含单个错误消息的响应,而在其他情况下,我们可能需要返回一个包含多个错误消息的响应。通过使用 ApiError 类,我们可以在应用程序中统一处理这些情况,并返回一个标准的错误响应格式。

结论

通过使用 ASP.NET Core 中间件和常用类,我们可以自定义 ASP.NET Core 应用程序中的响应格式,并标准化应用程序中的响应格式。这可以提高代码重用性,并使前端更加轻松地处理所有响应。在开发 ASP.NET Core 应用程序时,我们应该始终考虑使用中间件和常用类来提高代码的可读性、可维护性和可重用性。

原文链接:https://www.cnblogs.com/ke210/p/17298562.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:.net core 自定义规范响应的中间件 - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月18日

相关文章

  • C#组件系列 你值得拥有的一款Excel处理神器Spire.XLS

    C#组件系列你值得拥有的一款Excel处理神器Spire.XLS 什么是Spire.XLS Spire.XLS是一款专门针对Microsoft Excel读写的C#组件。它允许您在C#应用程序中读写Excel文件,可以方便的读取Excel文件中的内容,以及新建、编辑、保存Excel文件。 Spire.XLS的功能 Spire.XLS可以进行很多Excel文件…

    C# 2023年5月15日
    00
  • c# 抓取Web网页数据分析

    C# 抓取 Web 网页数据分析攻略 在使用 C# 抓取网页数据进行数据分析的过程中,主要需要做以下几个步骤: 发送 HTTP 请求,并获取网页 HTML 内容。 使用正则表达式或其他技术从 HTML 中提取需要的数据。 分析数据并进行处理,例如存储到数据库,生成报表,或进行可视化等操作。 下面将用示例说明这些步骤。 步骤一:发送 HTTP 请求并获取网页 …

    C# 2023年6月2日
    00
  • c#中值类型和引用类型的基础教程

    下面是关于“c#中值类型和引用类型的基础教程”的完整攻略: 概述 在C#中,变量可以分为两种类型:值类型和引用类型。这两种类型在内存中有不同的处理方式,因此在使用时需注意它们之间的差异。 值类型 值类型的变量直接存储其值,这意味着它们被存储在程序的栈中。栈内存是一种自动分配和释放的内存,通常用于存储函数参数和局部变量等短期数据。 C#中有多种内置的值类型,如…

    C# 2023年6月1日
    00
  • C#实现多个计时器记录不同定时时间

    实现多个计时器可以利用C#中的System.Timers.Timer类来完成。 步骤如下: 创建一个Dictionary<string, Timer>,用于存储多个计时器,其中键为计时器的名称,值为对应的Timer实例。 对于每个需要计时的任务,创建一个计时器并设置定时时间、事件处理程序等参数。 将计时器实例添加到Dictionary中,并指定一…

    C# 2023年6月1日
    00
  • C#中常量和只读变量的区别小结

    下面是详细的讲解“C#中常量和只读变量的区别小结”的完整攻略: 常量和只读变量的区别 在C#中,常量和只读变量都用来表示一些固定不变的值,但它们之间有明显的区别。 常量 常量在定义后就无法再次被修改。常量的值必须在编译时确定。常量使用const关键字来定义,如下所示: const int MaxValue = 100; 常量的名称必须以大写字母开头。在使用常…

    C# 2023年6月7日
    00
  • FreeSSL申请免费证书

    FreeSSL申请免费证书 FreeSSL 是一个免费证书和 SSL 证书管理平台。旨在为个人和小型企业提供免费 SSL 证书,以加强他们的网站和应用程序的安全性。与传统的 SSL 证书颁发机构不同,FreeSSL 使用自动化过程生成 SSL 证书,并提供一个管理面板,让用户可以轻松管理他们的证书和域名。 1.访问地址 https://freessl.cn/…

    C# 2023年5月8日
    00
  • asp.net下比较两个等长字符串是否含有完全相同字符(忽略字符顺序)

    要在ASP.NET下比较两个等长字符串是否含有完全相同的字符(忽略字符顺序),一种方法是对每个字符串进行排序,然后将结果进行比较。下面是具体的步骤。 第一步:定义比较函数 首先,我们需要定义一个比较函数。这个函数用于对字符串进行排序,并将排序结果作为函数的返回值。 public string SortString(string s) { char[] ca …

    C# 2023年6月7日
    00
  • c# WinForm 窗体之间传值的几种方式(小结)

    C# WinForm 窗体之间传值的几种方式(小结)是一篇关于C# WinForm中多个窗体之间进行数据传输的教程。下面,我将详细讲解这篇攻略的内容。 概述 该文主要讲解如何在 C# WinForm 中实现窗体之间的数据传输,我们知道在 Windows 系统下的许多应用程序中,除了主窗体以外可能会有一些其他的功能窗体,如登录窗体、设置窗体、信息弹窗等等,这些…

    C# 2023年6月7日
    00
合作推广
合作推广
分享本页
返回顶部