今年1月31日,在微軟的MVP 2015社區大講堂上,我給大家分享了一個演講:在ASP.NET應用中執行后台任務。其中介紹了三種技術的應用:QueueBackgroundWorkItem、Hangfire和Azure WebJobs。當時由於時間關系,對三者都是簡要介紹。今天想和大家着重介紹下Hangfire。
Hangfire是一個開源且商業免費使用的工具函數庫。可以讓你非常容易地在ASP.NET應用(也可以不在ASP.NET應用)中執行多種類型的后台任務,而無需自行定制開發和管理基於Windows Service后台任務執行器。且任務信息可以被持久保存。內置提供集成化的控制台。
通過Nuget就可以在你的應用程序中安裝Hangfire:Install-Package Hangfire
Hangfire的具有如下特性和有點:
- 支持基於隊列的任務處理:任務執行不是同步的,而是放到一個持久化隊列中,以便馬上把請求控制權返回給調用者。使用方法:BackgroundJob.Enqueue(() => Console.WriteLine("Simple!"));
- 延遲任務執行:不是馬上調用方法,而是設定一個未來時間點再來執行。使用方法:BackgroundJob.Schedule(() => Console.WriteLine("Reliable!"), TimeSpan.FromDays(7));
- 循環任務執行:只需要簡單的一行代碼就可以添加重復執行的任務,其內置了常見的時間循環模式,也可以基於CRON表達式來設定復雜的模式。使用方法:RecurringJob.AddOrUpdate(() => Console.WriteLine("Transparent!"), Cron.Daily);
- 持久化保存任務、隊列、統計信息:默認使用SQL Server,也可以配合消息隊列來降低隊列處理延遲,或配置使用Redis來獲得更好的性能表現
- 內置自動重試機制:可以設定重試次數,還可以手動在控制台重啟任務
- 除了調用靜態方法外還支持實例方法
- 能夠捕獲多語言狀態:即可以把調用者的Thread.CurrentCulture和Thread.CurrentUICulture信息同任務持久保存在一起,以便任務執行的時候多語言信息是一致的
- 支持任務取消:使用CancellationToken這樣的機制來處理任務取消邏輯
- 支持IoC容器:目前支持Ninject和Autofac比較常用的開源IoC容器
- 支持Web集群:可以在一台或多台機器上運行多個Hangfire實例以便實現冗余備份
- 支持多隊列:同一個Hangfire實例可以支持多個隊列,以便更好的控制任務的執行方式
- 並發級別的控制:默認是處理器數量的5倍工作行程,當然也可以自己設定
- 具備很好的擴展性:有很多擴展點來控制持久存儲方式、IoC容器支持等
為什么要使用Hangfire這樣的函數庫呢?我覺得好處有如下幾個方面:
- 開發簡單:無需自己額外做開發,就可以實現任務的隊列執行、延遲執行和重復執行
- 部署簡單:可以同主ASP.NET應用部署在一起,測試和維護都相對簡單
- 遷移簡單:由於宿主不僅限於ASP.NET,那么未來可以非常容易的把任務執行器放到其他地方(需要改變的就是在其他宿主中啟動Hangfire服務器)
- 擴展簡單:由於開源且有很多擴展點,在現有插件都不滿足自己需要的情況下能夠容易的進行擴展
之前我把Hangfire運用到兩種情況下:
- 后台長時間的科學計算:這樣就可以讓請求馬上返回給客戶端,后台完成長時間計算后,用SignalR實時提醒用戶
- 后台群發郵件:通過延遲和循環任務分批通過SendCloud這樣的服務發送群發郵件
當然,Hangfire的應用場景還很多,比如在后台處理電商賣家的訂單。
MVP 2015社區大講堂演講的PPT在此:MVP 2015社區大講堂之:在ASP.NET應用中執行后台任務