主頁 > .NET開發 > ASP.NET Core 3.x啟動時運行異步任務(一)

ASP.NET Core 3.x啟動時運行異步任務(一)

2020-09-16 15:21:33 .NET開發

這是一個大的題目,需要用幾篇文章來說清楚,這是第一篇,

一、前言

在我們的專案中,有時候我們需要在應用程式啟動前執行一些一次性的邏輯,比方說:驗證配置的正確性、填充快取、或者運行資料庫清理/遷移等,

如何合理、有效、優雅地完成這個任務,是這個文章討論的主要內容,

要實作這樣一個功能,其實我們有幾個選擇:

  1. 使用IStartupFilter運行同步任務,這是一個內置的解決方案,可以通過一些設定和技巧來運行異步任務;
  2. 使用IStartupFilterIApplicationLifetime事件來運行異步任務,這是一個可選的方案,但有不足,我們會在后面講;
  3. 使用IHostedService,在不阻塞應用啟動的情況下,運行一些一次性的任務;(關于這個內容,我在前一篇文章ASP.NET Core 3.x控制IHostedService啟動順序淺探中有涉及到一部分內容)
  4. Program.cs中運行異步任務,在大多數情況下,從代碼的復雜度到效率上,這都是一個比較好的選擇,

    為防止非授權轉發,這兒給出本文的原文鏈接:https://www.cnblogs.com/tiger-wang/p/13673046.html

先提個問題:為什么要在應用啟動時運行任務?

二、為什么要在應用啟動時運行任務?

在應用啟動并開始請求服務之前,很多時候需要運行各種初始化作業,

一個ASP.NET應用啟動時,需要完成很多事,例如:

  • 確定當前的宿主環境
  • 加載appsetting.json配置和環境變數
  • 配置并創建依賴注入的容器
  • 配置中間件管道

這是應用啟動時要完成的引導內容,

在完成這些內容,運行WebHost并開始監聽請求之前,還會有一些一次性任務需要啟動,例如:

  • 檢查強型別配置的有效性
  • 填充或恢復快取
  • 資料庫清理/遷移(通常來說這不是個好主意,但很多時候沒有別的辦法)

當然,有些任務也不是一定要在開始監聽請求之前運行,這要看具體的運行任務的架構,一般來說,如果快取處理的完善,是不需要提前啟動的,當然,清理/遷移資料庫,是必須放在服務啟動之前,

在微軟官網上,有一個例子是資料保護子系統,用于即時加密(cookie、防偽令牌等),這個就必須在應用監聽請求之前完成初始化并加載,這個例子使用了IStartupFilter

三、使用IStartupFilter運行同步任務

IStartupFilters作為配置中間件管道的一部分,通常在Startup.Configure()中運行,它允許我們定制應用的中間件管道,處理我們希望進行的所有任務,

看一個簡單的例子:

public class AutoRequestServicesStartupFilter : IStartupFilter
{
    public Action<IApplicationBuilder> Configure(Action<IApplicationBuilder> next)
    {
        return builder =>
        {
            builder.UseMiddleware<RequestServicesContainerMiddleware>();
            next(builder);
        };
    }
}

IStartupFilter提供了一種可能,在依賴注入容器配置完成之后、應用程式啟動之前運行一些代碼,因此,我們可以在IStartupFilters中直接使用依賴注入,這表示我們可以運行有關系統的任何代碼,在前邊提到的微軟官網的例子中,就是創建了一個基于IStartupFiltersDataProtectionStartupFilter來初始化資料保護子系統,

此外,IStartupFilter允許我們通過向依賴注入容器注冊服務來增加要執行的任務,這是一個很有用的特性,表示我們可以注冊一個在應用啟動時運行的任務,而不需要顯式的呼叫,

但是,這兒有個問題,IStartupFilters通常運行的是同步的任務,看一下上面的代碼,Configure()方法不回傳任務,當然,我們硬要使用異步也是可以的,但一般來說,這不算個好主意,原因我后面會寫,

寫到這兒,如果對ASP.NET Core架構熟悉,就會引出另一個問題:為什么不用健康檢查來確認一次性任務的執行結果?

四、為什么不用健康檢查?

運行健康檢查,是ASP.NET Core 2.2新引入的一個特性,允許查詢通過API(HTTP Endpoint)公開的應用的健康狀況,當應用部署在Kubernetes,或反向代理HAProxyNginx后面時,可以提供給代理用來檢測應用是否準備好開始提供服務,

我們可以使用健康檢查來確保應用所有必需的一次性任務完成之前不會開始監聽服務,

但是,這種方式會有一點問題,

WebHostKestrel本身會在一次性任務執行前啟動,當然,這時他們還不會接收和處理服務請求,但仍然引出了一些問題:

首先是增加了代碼的復雜性,除了一次性任務的代碼外,還要增加健康檢查來測驗任務是否完成,并同步和保持任務的狀態;其次,如果任務失敗了,應用程式的健康檢查將會讓應用后續的任務無法繼續執行,合理的流程是:應用應該立即失敗回傳,

這兒主要的原因是:健康檢查沒有定義如何實際運行任務,而只是定義了任務是否成功完成,相對來說,這種狀態機制比較單一,在一些簡單的任務中可能適用,但不能全面覆寫一次性任務的全部場景,

五、運行異步任務

前邊寫了一些不太完美的方法,

現在,我們開始進入運行異步方法的一些步驟,當然,運行異步也會有幾種方式,適用性上會有一定的區別,

方式1:使用IStartupFilter

前邊說過,使用IStartupFilter時,執行的是同步任務,所以,我們可以通過GetAwater().GetResult()來呼叫異步,

我們拿資料遷移來舉個例子,在EF Core中,通過myDBContext.database.migrateasync()在運行時進行資料庫遷移,其中,myDBContext是應用程式中DBContext的一個實體,

public class MigratorStartupFilter: IStartupFilter
{
    private readonly IServiceProvider _serviceProvider;
    public MigratorStartupFilter(IServiceProvider serviceProvider)
    
{
        _serviceProvider = serviceProvider;
    }

    public Action<IApplicationBuilder> Configure(Action<IApplicationBuilder> next)
    {
        using(var scope = _seviceProvider.CreateScope())
        {
            var myDbContext = scope.ServiceProvider.GetRequiredService<MyDbContext>();

            myDbContext.Database.MigrateAsync()
                .GetAwaiter()
                .GetResult();
        }

        return next;
    }
}

通常,GetAwaiter().GetResult()要注意避免死鎖的問題,但這兒可能不需要,因為這個代碼只在啟動時運行,這時候還沒有需要處理的請求,所以不太會死鎖,

只能說,這樣可以用,不過習慣上我會避免這么做,

方式2:使用IApplicationLifetime事件

這是另一個選擇,可以通過IApplicationLifetime事件,在應用啟動和關閉時接收通知,處理任務,

但這個方式也有局限性,

首先,IApplicationLifetime使用cancellationtoken來注冊回呼,也就是說,這又是一個同步方式,又需要使用GetAwaiter().GetResult()來呼叫異步,

其次,ApplicationStarted事件是在WebHost啟動之后才會觸發,因此異步任務也是在應用開始監聽請求后才運行,

方式3:使用IHostedService

IHostedService可以讓ASP.NET Core應用在后臺執行長時間的任務,

一般來說,IHostedService用在周期性任務、訊息傳遞等任務上,但實際上它并不限于運行這些任務,在ASP.NET Core 3.x上,WebHost本身也是建立在IHostedService上的,

而且,IHostedService本身就是異步的,它提供了StartAsyncStopAsync

這種方式下,我們的代碼會是這樣:

public class MigratorHostedService: IHostedService
{
    private readonly IServiceProvider _serviceProvider;
    public MigratorStartupFilter(IServiceProvider serviceProvider)
    
{
        _serviceProvider = serviceProvider;
    }

    public async Task StartAsync(CancellationToken cancellationToken)
    
{
        using(var scope = _seviceProvider.CreateScope())
        {
            var myDbContext = scope.ServiceProvider.GetRequiredService<MyDbContext>();

            await myDbContext.Database.MigrateAsync();
        }
    }

    public Task StopAsync(CancellationToken cancellationToken)
    
{
        return Task.CompletedTask;
    }
}

根據例子可以看出,IHostedService可以直接運行異步任務,

但是,IHostedService也有局限性,從微軟官網的說明來看,IHostedService實作期望StartAsync能相對較快的回傳,對于后臺任務,傾向于異步啟動,但主要任務在啟動后執行,

在上面這個例子中,資料遷移本身不是問題,但這個長時任務會阻止其它`IHostedService啟動和運行,而且,應用會在IHostedService完成資料遷移前開始監聽并回應請求,這是一個嚴重的問題,

方式4:在Program.cs中運行

上面三個方式,都可以解決啟動時運行異步任務的問題,但都不夠完美,要么要求使用同步(異步轉同步可以用,但有隱藏問題),要么不能阻止應用啟動,會造成應用啟動完成后,可能異步任務還未完成的情況,

我在前邊的博文中寫到過關于Program.cs中運行IHostedService的方式,具體可以去看ASP.NET Core 3.x控制IHostedService啟動順序淺探

看一下Program.cs的默認代碼:

public class Program
{

    public static void Main(string[] args)
    
{
        CreateWebHostBuilder(args).Build().Run();
    }

    public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
        WebHost.CreateDefaultBuilder(args)
            .UseStartup<Startup>();
}

Build()創建WebHost之后,呼叫Run()之前,完全可以加入我們需要的代碼,同時,C# 7.1后主函式可以改為異步運行,

因此,我們可以在這兒做些文章:

public class Program
{

    public static async Task Main(string[] args)
    
{
        IWebHost webHost = CreateWebHostBuilder(args).Build();

        using (var scope = webHost.Services.CreateScope())
        {
            var myDbContext = scope.ServiceProvider.GetRequiredService<MyDbContext>();

            await myDbContext.Database.MigrateAsync();
        }

        await webHost.RunAsync();
    }

    public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
        WebHost.CreateDefaultBuilder(args)
            .UseStartup<Startup>();
}

這個方案的好處是:

  • 這是真正的異步;
  • 任務完成后,應用程式才可以監聽并接受請求;
  • 此時已經構建了依賴注入容器,所以可以創建服務;

當然,同樣也會有不足:這兒只是構建了DI容器,但并沒有建立管道(管道在Run()RunAsync()后才建立,然后是IStartupFilters執行,再然后是應用程式啟動),因此異步任務不能使用管道、IStartupFilters中的配置,不過,這種需求的情況很少,

六、總結

這個部分牽扯到的框架內容比較多,

我們從應用啟動時異步運行任務開始,說到了必要性,也說到了幾種解決方法,及各自的優缺點,

下一篇文章,我會用一些具體的例子,來說清楚這個方式的具體使用,敬請關注,

(未完待續)

 

 


 

微信公眾號:老王Plus

掃描二維碼,關注個人公眾號,可以第一時間得到最新的個人文章和內容推送

本文著作權歸作者所有,轉載請保留此宣告和原文鏈接

轉載請註明出處,本文鏈接:https://www.uj5u.com/net/58066.html

標籤:.NET Core

上一篇:怎么限制上傳檔案型別

下一篇:ASP.NET Core 設定運行埠,啟動多個服務實體

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • WebAPI簡介

    Web體系結構: 有三個核心:資源(resource),URL(統一資源識別符號)和表示 他們的關系是這樣的:一個資源由一個URL進行標識,HTTP客戶端使用URL定位資源,表示是從資源回傳資料,媒體型別是資源回傳的資料格式。 接下來我們說下HTTP. HTTP協議的系統是一種無狀態的方式,使用請求/ ......

    uj5u.com 2020-09-09 22:07:47 more
  • asp.net core 3.1 入口:Program.cs中的Main函式

    本文分析Program.cs 中Main()函式中代碼的運行順序分析asp.net core程式的啟動,重點不是剖析原始碼,而是理清程式開始時執行的順序。到呼叫了哪些實體,哪些法方。asp.net core 3.1 的程式入口在專案Program.cs檔案里,如下。ususing System; us ......

    uj5u.com 2020-09-09 22:07:49 more
  • asp.net網站作為websocket服務端的應用該如何寫

    最近被websocket的一個問題困擾了很久,有一個需求是在web網站中搭建websocket服務。客戶端通過網頁與服務器建立連接,然后服務器根據ip給客戶端網頁發送資訊。 其實,這個需求并不難,只是剛開始對websocket的內容不太了解。上網搜索了一下,有通過asp.net core 實作的、有 ......

    uj5u.com 2020-09-09 22:08:02 more
  • ASP.NET 開源匯入匯出庫Magicodes.IE Docker中使用

    Magicodes.IE在Docker中使用 更新歷史 2019.02.13 【Nuget】版本更新到2.0.2 【匯入】修復單列匯入的Bug,單元測驗“OneColumnImporter_Test”。問題見(https://github.com/dotnetcore/Magicodes.IE/is ......

    uj5u.com 2020-09-09 22:08:05 more
  • 在webform中使用ajax

    如果你用過Asp.net webform, 說明你也算是.NET 開發的老兵了。WEBform應該是2011 2013左右,當時還用visual studio 2005、 visual studio 2008。后來基本都用的是MVC。 如果是新開發的專案,估計沒人會用webform技術。但是有些舊版 ......

    uj5u.com 2020-09-09 22:08:50 more
  • iis添加asp.net網站,訪問提示:由于擴展配置問題而無法提供您請求的

    今天在iis服務器配置asp.net網站,遇到一個問題,記錄一下: 問題:由于擴展配置問題而無法提供您請求的頁面。如果該頁面是腳本,請添加處理程式。如果應下載檔案,請添加 MIME 映射。 WindowServer2012服務器,添加角色安裝完.netframework和iis之后,運行aspx頁面 ......

    uj5u.com 2020-09-09 22:10:00 more
  • WebAPI-處理架構

    帶著問題去思考,大家好! 問題1:HTTP請求和回傳相應的HTTP回應資訊之間發生了什么? 1:首先是最底層,托管層,位于WebAPI和底層HTTP堆疊之間 2:其次是 訊息處理程式管道層,這里比如日志和快取。OWIN的參考是將訊息處理程式管道的一些功能下移到堆疊下端的OWIN中間件了。 3:控制器處理 ......

    uj5u.com 2020-09-09 22:11:13 more
  • 微信門戶開發框架-使用指導說明書

    微信門戶應用管理系統,采用基于 MVC + Bootstrap + Ajax + Enterprise Library的技術路線,界面層采用Boostrap + Metronic組合的前端框架,資料訪問層支持Oracle、SQLServer、MySQL、PostgreSQL等資料庫。框架以MVC5,... ......

    uj5u.com 2020-09-09 22:15:18 more
  • WebAPI-HTTP編程模型

    帶著問題去思考,大家好!它是什么?它包含什么?它能干什么? 訊息 HTTP編程模型的核心就是訊息抽象,表示為:HttPRequestMessage,HttpResponseMessage.用于客戶端和服務端之間交換請求和回應訊息。 HttpMethod類包含了一組靜態屬性: private stat ......

    uj5u.com 2020-09-09 22:15:23 more
  • 部署WebApi隨筆

    一、跨域 NuGet參考Microsoft.AspNet.WebApi.Cors WebApiConfig.cs中配置: // Web API 配置和服務 config.EnableCors(new EnableCorsAttribute("*", "*", "*")); 二、清除默認回傳XML格式 ......

    uj5u.com 2020-09-09 22:15:48 more
最新发布
  • C#多執行緒學習(二) 如何操縱一個執行緒

    <a href="https://www.cnblogs.com/x-zhi/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2943582/20220801082530.png" alt="" /></...

    uj5u.com 2023-04-19 09:17:20 more
  • C#多執行緒學習(二) 如何操縱一個執行緒

    C#多執行緒學習(二) 如何操縱一個執行緒 執行緒學習第一篇:C#多執行緒學習(一) 多執行緒的相關概念 下面我們就動手來創建一個執行緒,使用Thread類創建執行緒時,只需提供執行緒入口即可。(執行緒入口使程式知道該讓這個執行緒干什么事) 在C#中,執行緒入口是通過ThreadStart代理(delegate)來提供的 ......

    uj5u.com 2023-04-19 09:16:49 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    <a href="https://www.cnblogs.com/huangxincheng/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/214741/20200614104537.png" alt="" /&g...

    uj5u.com 2023-04-18 08:39:04 more
  • 記一次 .NET某醫療器械清洗系統 卡死分析

    一:背景 1. 講故事 前段時間協助訓練營里的一位朋友分析了一個程式卡死的問題,回過頭來看這個案例比較經典,這篇稍微整理一下供后來者少踩坑吧。 二:WinDbg 分析 1. 為什么會卡死 因為是表單程式,理所當然就是看主執行緒此時正在做什么? 可以用 ~0s ; k 看一下便知。 0:000> k # ......

    uj5u.com 2023-04-18 08:33:10 more
  • SignalR, No Connection with that ID,IIS

    <a href="https://www.cnblogs.com/smartstar/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/u36196.jpg" alt="" /></a>...

    uj5u.com 2023-03-30 17:21:52 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:15:33 more
  • 一次對pool的誤用導致的.net頻繁gc的診斷分析

    <a href="https://www.cnblogs.com/dotnet-diagnostic/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/3115652/20230225090434.png" alt=""...

    uj5u.com 2023-03-28 10:13:31 more
  • C#遍歷指定檔案夾中所有檔案的3種方法

    <a href="https://www.cnblogs.com/xbhp/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/957602/20230310105611.png" alt="" /></a&...

    uj5u.com 2023-03-27 14:46:55 more
  • C#/VB.NET:如何將PDF轉為PDF/A

    <a href="https://www.cnblogs.com/Carina-baby/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/2859233/20220427162558.png" alt="" />...

    uj5u.com 2023-03-27 14:46:35 more
  • 武裝你的WEBAPI-OData聚合查詢

    <a href="https://www.cnblogs.com/podolski/" target="_blank"><img width="48" height="48" class="pfs" src="https://pic.cnblogs.com/face/616093/20140323000327.png" alt="" /><...

    uj5u.com 2023-03-27 14:46:16 more