.NET Core使用 CancellationToken 取消API請求

chester·chen發表於2024-03-17

您是否曾經訪問過一個網站,它需要很長時間載入,最終你敲擊 F5 重新載入頁面。

即使使用者重新整理了瀏覽器取消了原始請求,而對於伺服器來說,API也不會知道它正在計算的值將在結束時被丟棄,重新整理五次,伺服器將觸發 5 個請求。

為了解決這個問題,ASP.NET Core 為 Web 伺服器提供了一種機制,就是CancellationToken.

使用者取消請求時,你可以使用HttpContext.RequestAborted訪問,您也可以使用依賴注入將其自動注入到您的操作中。

長時間執行的任務請求

現在我們假設您有一個 API 操作,在向使用者傳送響應之前可能需要一些時間才能完成。

在處理該操作時,使用者可以直接取消請求,或重新整理頁面(這會有效地取消原始請求,並啟動新請求)。

[HttpGet(Name = "get")]
public async Task<string> GetAsync()
{
    try
    {
        _logger.LogInformation("request in");
        await Task.Delay(5 * 1000);
        _logger.LogInformation("request end");
    }
    catch (Exception ex)
    {
        _logger.LogInformation("request ex");
    }
    return "ok";
}

如果使用者在請求中途重新整理瀏覽器,那麼瀏覽器永遠不會收到第一個請求的響應,但在server端可以看到,操作方法執行完成兩次。

這是否是正確將取決於您的應用程式。

如果請求修改某些業務的狀態,那麼您可能不希望在方法中途停止執行。如果請求沒有副作用,那麼您可能希望儘快停止(可能很昂貴)操作。

使用者取消請求時,你可以使用HttpContext.RequestAborted訪問,您也可以使用依賴注入將其自動注入到您的操作中。

CancellationTokens取消不必要的請求

以下程式碼顯示瞭如何透過將 CancellationTokenSource 注入到操作方法中,並透過其取消不必要的操作。

[HttpGet(Name = "get")]
public async Task<string> GetAsync(CancellationToken cancellationToken)
{
    try
    {
        _logger.LogInformation("request in");
        await Task.Delay(5 * 1000,cancellationToken);
        _logger.LogInformation("request end");
    }
    catch (Exception ex)
    {
        _logger.LogInformation("request ex");
    }
    return "ok";
}

透過這個改變,我們可以再次測試我們的場景。

我們發出一個初始請求,然後我們重新載入頁面。正如您從下面的日誌中看到的,第一個請求不會繼續執行。

使用者重新整理瀏覽器取消請求後不久,原始請求就會中止,並TaskCancelledException透過 API 過濾器管道傳播回來,並備份中介軟體管道。

根據您的場景,您可能能夠依靠此類框架方法來檢查 的狀態CancellationToken,或者您可能需要自己監視取消請求。

過濾器捕獲異常

您可以透過以上try catch 捕獲,或者透過一個過濾器統一監視此異常。

public class OperationCancelledExceptionFilter : ExceptionFilterAttribute
{
    private readonly ILogger _logger;

    public OperationCancelledExceptionFilter(ILoggerFactory loggerFactory)
    {
        _logger = loggerFactory.CreateLogger<OperationCancelledExceptionFilter>();
    }
    public override void OnException(ExceptionContext context)
    {
        if (context.Exception is OperationCanceledException)
        {
            _logger.LogInformation("Request was cancelled");
            context.ExceptionHandled = true;
            context.Result = new StatusCodeResult(400);
        }
    }
}


builder.Services.AddControllers(options =>
{
    options.Filters.Add<OperationCancelledExceptionFilter>();
});

相關文章