一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

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

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

服務器之家 - 編程語言 - ASP.NET教程 - 由ASP.NET Core讀取Response.Body引發的思考

由ASP.NET Core讀取Response.Body引發的思考

2023-05-08 00:02未知服務器之家 ASP.NET教程

前言 ????前幾天有群友在群里問如何在我之前的文章《ASP.NET Core WebApi返回結果統一包裝實踐》的時候有點疑問,主要的疑問點就是關于Respouse的讀取的問題。在之前的文章《深入探究ASP.NET Core讀取Request.Body的正確方式》曾分析過

前言

????前幾天有群友在群里問如何在我之前的文章《ASP.NET Core WebApi返回結果統一包裝實踐》的時候有點疑問,主要的疑問點就是關于Respouse的讀取的問題。在之前的文章《深入探究ASP.NET Core讀取Request.Body的正確方式》曾分析過關于Request的讀取問題,需要讀取Respouse的場景同樣經常遇到,比如讀取輸出信息或者包裝一下輸出結果等。無獨有偶Respouse的讀取同樣存在類似的問題,本文我們便來分析一下如何進行Response的Body讀取。

使用方式

我們在日常的使用中是如何讀取流呢?很簡單,直接使用StreamReader去讀取,方式如下

public override void OnResultExecuted(ResultExecutedContext context)
{
    //操作流之前恢復一下操作位
    context.HttpContext.Response.Body.Position = 0;

    StreamReader stream = new StreamReader(context.HttpContext.Response.Body);
    string body = stream.ReadToEnd();
    _logger.LogInformation("body content:" + body);

    context.HttpContext.Response.Body.Position = 0;
    base.OnResultExecuted(context);
}

代碼很簡單,直接讀取即可,可是這樣讀取是有問題的會拋出異常System.ArgumentException:“Stream was not readable.”異常信息就是的意思是當前Stream不可讀,也就是Respouse的Body是不可以被讀取的。關于StreamReader到底和Stream有啥關聯,我們在之前的文章深入探究ASP.NET Core讀取Request.Body的正確方式一文中有過源碼分析,這里就不在贅述了,有興趣的同學可以自行翻閱,強烈建議在閱讀本文之前可以看一下那篇文章,方便更容易了解。
如何解決上面的問題呢?方式也很簡單,比如你想在你的程序中保證Response的Body都是可讀的,你可以定義一個中間件解決這個問題。

public static IApplicationBuilder UseResponseBodyRead(this IApplicationBuilder app)
{
    return app.Use(async (context, next) =>
    {
        //獲取原始的Response Body
        var originalResponseBody = context.Response.Body;
        try
        {
            //聲明一個MemoryStream替換Response Body
            using var swapStream = new MemoryStream();
            context.Response.Body = swapStream;
            await next(context);
            //重置標識位
            context.Response.Body.Seek(0, SeekOrigin.Begin);
            //把替換后的Response Body復制到原始的Response Body
            await swapStream.CopyToAsync(originalResponseBody);
        }
        finally
        {
            //無論異常與否都要把原始的Body給切換回來
            context.Response.Body = originalResponseBody;
        }
    });
}

本質就是先用一個可操作的Stream比如咱們這里的MemoryStream替換默認的ResponseBody,讓后續對ResponseBody的操作都是針對新的ResponseBody進行操作,完成之后把替換后的ResponseBody復制到原始的ResponseBody。最終無論異常與否都要把原始的Body給切換回來。需要注意的是,這個中間件的位置盡量要放在比較靠前的位置注冊,至少也要保證在你所有要操作ResponseBody之前的位置注冊。如下所示

var app = builder.Build();
app.UseResponseBodyRead();

源碼探究

通過上面我們了解到了ResponseBody是不可以被讀取的,至于為什么呢,這個我們需要通過相關源碼了解一下。通過HttpContext類的源碼我們可以看到相關定義

public abstract class HttpContext
{
    public abstract HttpResponse Response { get; }
}

這里看到HttpContext本身是個抽象類,看一下它的屬性HttpResponse類的定義也是一個抽象類

public abstract class HttpResponse
{
}

由上面可知Response屬性是抽象的,所以抽象類HttpResponse必然包含一個子類去實現它,否則沒辦法直接操作相關方法。這里我們介紹一個網站https://source.dot.net用它可以更輕松的閱讀微軟類庫的源碼,比如CLR、ASP.NET Core、EF Core等等,雙擊一個類或者屬性方法可以查找引用和定義它們的地方,非常方便,它的源碼都是最新版本的,來源就是GitHub上的相關倉庫。找到實例化HttpResponse的為位置在HttpContext的子類DefaultHttpContext類中[點擊查看源碼

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 国模孕妇季玥全部人体写真 | 久久久伊人影院 | 女人c交zzzooo在线观看 | 亚洲精品视频在线免费 | 久久这里只有精品国产精品99 | 521色香蕉网在线观看免费 | 92国产福利久久青青草原 | 双性小说肉 | www.羞羞答答 | 398av影院视频在线 | 日韩精品一区二区三区毛片 | 亚洲男人第一天堂 | 91制片厂制作果冻传媒破解 | 日韩在线免费 | 朝鲜女人free性hu | 4hc44四虎永久地址链接 | jizzjizzjⅰzz亚洲美女 | 久久久乱码精品亚洲日韩 | 欧美yyy| 亚洲波多野结衣日韩在线 | 国产小视频在线免费 | 国产亚洲高清国产拍精品 | 国产精品女同久久免费观看 | 精品欧美一区二区精品久久 | 亚洲2023无矿砖码砖区 | 苍井空50分钟无码 | 国产精品调教 | 毛片免费的 | 国产人人草 | 免费网址在线观看入口推荐 | 精品国产91久久久久久久 | caoporen97免费公开视频 | 国产一区二区播放 | 久久久久久久久女黄 | 2022最新国产在线不卡a | 久99视频精品免费观看福利 | 日本三不卡 | а天堂中文最新版在线官网视频 | 国产精品制服丝袜白丝www | 情趣内衣在线观看 | 插入粉嫩 |