在 ASP.NET Core 程式啟動前運行你的代碼

来源:https://www.cnblogs.com/danvic712/archive/2020/01/20/run-your-code-before-asp-net-core-program-start.html
-Advertisement-
Play Games

一、前言 在進行 Web 項目開發的過程中,可能會存在一些需要經常訪問的靜態數據,針對這種在程式運行過程中可能幾乎不會發生變化的數據,我們可以嘗試在程式運行前寫入到緩存中,這樣在系統後續使用時就可以直接從緩存中進行獲取,從而減緩因為頻繁讀取這些靜態數據造成的應用資料庫伺服器的巨大承載壓力。 既然需要 ...


一、前言

在進行 Web 項目開發的過程中,可能會存在一些需要經常訪問的靜態數據,針對這種在程式運行過程中可能幾乎不會發生變化的數據,我們可以嘗試在程式運行前寫入到緩存中,這樣在系統後續使用時就可以直接從緩存中進行獲取,從而減緩因為頻繁讀取這些靜態數據造成的應用資料庫伺服器的巨大承載壓力。

既然需要在程式運行前將靜態數據寫入到緩存中,毫無疑問我們需要在程式運行前執行一些自定義功能的代碼,那麼在本章中,我將會介紹如何在 ASP.NET Core 項目中,實現在程式啟動前執行某些特定功能的代碼。

 

二、Step by Step

1、先說結論

因為這一篇文章更多的是在說明我在解決這個問題時的一步步思考,並沒有涉及到代碼的編寫,所以下麵的內容可能對你的幫助並不是很大,所以這裡提前將實現的方式告訴大家。對於這個問題來說,只需要將我們想要執行的代碼放到下麵代碼中註釋所在的位置,即可實現我們的需求。

public class Program
{
  public static void Main(string[] args)
  {
    var host = CreateHostBuilder(args).Build();
        
    // do what you want
        
    host.Run();
  }

    public static IHostBuilder CreateHostBuilder(string[] args) =>
        Host.CreateDefaultBuilder(args)
            .ConfigureWebHostDefaults(webBuilder =>
            {
                webBuilder.UseStartup<Startup>();
            });
}

2、前車之鑒

在嘗試如何在 ASP.NET Core 中實現這一功能需求前,我們可以看看在 .NET Framework 中如何實現這一功能,是不是可以對我們在後續的功能實現中提供某些借鑒。

對於採用 .NET Framework 的應用程式來說,項目創建後會生成一個 Global.asax 文件,在這個類文件中存在著 Application_Start 這樣的一個方法,而 Application_Start 這個方法實際上是在當應用程式接收到第一個 HTTP 請求時觸發,也就是說,當系統運行後第一次接收到用戶的請求,就會觸發 Application_Start 中的代碼邏輯,後續不管再接收到多少的請求,都不會再觸發該方法。

例如在這個基於 .NET Framework 構建的 MVC 項目模板中,在程式運行前需要執行註冊路由信息、註冊過濾器、註冊使用 bundle 壓縮後的 js、css 文件等等。

但是在 ASP.NET Core 項目中,並沒有原生存在這樣的方法,那麼我們如何在 ASP.NET Core 應用中自己動手實現類似的功能呢?

3、後事之師

瞭解了在之前版本中的實現方式,現在我們仔細看看 Application_Start 這個方法中執行的每行代碼的功能,是不是特別像我們在 ASP.NET Core 項目中使用的各種中間件?

然而,如果你有使用過 ASP.NET Core 後就會知道,ASP.NET Core 中的中間件是會在每次請求時都會觸發的,雖然我們可以在我們自定義的中間件中設置緩存中不存在數據就寫入,存在就直接跳過的代碼邏輯,但是既然除了第一次訪問時才會真正執行該中間件的功能,後面完全用不到,因此,對於我這種略微強迫症的童鞋來說,這個真的不能忍。。。

既然中間件不可以,而我們需要的僅僅是只運行一次,提到 .NET Core,不知道你的第一印象是什麼,對於我個人來說,無處不在的依賴註入,可能是我在 18 年開始學習 .NET Core 時的第一印象。我們知道,對於 .NET Core 中原生的依賴註入組件,存在著三種生命周期:Singleton、Scoped 以及 Transient,對於這三種生命周期的具體解釋,還是推薦博客園裡蔣金楠老師的一篇文章(電梯直達)。

對於採用 Singleton 方式註入的服務來說,因為是一種類似於全局單例的形式,不管後續從何處進行訪問,都會訪問的是同一個實例,那麼,這裡是不是就可以在此基礎上實現我們的需求了呢?

很不幸,這裡其實是有個很嚴重的邏輯上的問題的,依賴註入最終的目的是為了實現將我們定義的服務契約與實現進行解耦,實現服務的消費者只需要告訴依賴註入容器自己所需要服務的類型(服務介面 or 抽象服務類),就能自動得到與之匹配的服務實例。

簡單點說就是,消費方要告訴服務提供方你要開始使用某個服務了,我才能給你提供對應的服務,就像我們去飯店吃飯,在點了菜後,沒有必要關心廚師是用天然氣 or 煤氣給你燒的菜,但是能不能上菜的關鍵在於我們有沒有點菜。因此,這個問題最終還是落在了我們應該在程式中的什麼地方去調用我們設定好的方法。

繞了一圈,似乎我們的想法越來越偏,離我們想要實現的越來越遠,既然路偏了,那就直接回到起點吧,拋棄我們在 .NET Framework 項目中的經驗,重新從 ASP.NET Core 項目的啟動流程開始看起。

在 ASP.NET Core 應用的啟動過程中存在著兩個非常重要的對象,對應到我們採用的 ASP.NET Core 3.X 的項目中則是 Host 以及 HostBuilder。這裡的 Host 就是承載我們 Web 應用運行的載體,而 HostBuilder 則是用來構建 Host 對象的。

PS:因為 ASP.NET Core 3.0 開始加入了 對於 gRPC 框架和 Windows Service 的支持,同時為了與其它非 Web 伺服器方案進行集成,因此將原來的 WebHost 和 WebHostBuilder 替換成了新的通用主機(generic-host)配置的模式 。當然,在 3.X 版本你還是可以使用 WebHost 和 WebHostBuilder 的,不過當然是不推薦的。

因為對於 ASP.NET Core 應用程式來說,本質上其實只是一個控制台應用,所以現在我們來看看對於一個控制台應用中最重要的文件:Program.cs, Program 類中的代碼如下所示。

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

    public static IHostBuilder CreateHostBuilder(string[] args) =>
        Host.CreateDefaultBuilder(args)
            .ConfigureWebHostDefaults(webBuilder =>
            {
                webBuilder.UseStartup<Startup>();
            });
}

代碼很少,功能也很簡單,簡單來說,在 Main 方法中構建 HostBuilder 對象,然後去運行它,達到啟動我們 Web 應用宿主的目的。

當然,在構建 HostBuilder 對象的過程中,會配置 Kestrel 伺服器,會設置 ContentRoot,會載入配置文件等等一系列的動作,因為自己水平太次,嘗試了一下,還是解釋不好,如果你想要深入瞭解的話,建議配合博客園裡面的這兩篇文章一起食用(200行代碼,7個對象——讓你瞭解 ASP.NET Core 框架的本質ASP.NET Core 2.0 : 七.一張圖看透啟動背後的秘密)。雖然參考文章中都是基於 ASP.NET Core 2.X 版本進行解釋說明的,但其實最終的差異不是很大。

不知你是否找到了這個類中對於我們最重要的一點,在 Main 方法中,我們是先構建、再去運行,因此,我們是不是可以在構建完成後,先等一等,把我們想要實現的功能先調用了,再去運行我們的程式。嗯,讓我們改造下 Main 方法中的代碼。

public class Program
{
    public static void Main(string[] args)
    {
    var host = CreateHostBuilder(args).Build();

    // Get logger
    //
    var logger = host.Services.GetRequiredService<ILogger<Program>>();
    logger.LogInformation("haha, I ran before web host starting");

    host.Run();
  }

    public static IHostBuilder CreateHostBuilder(string[] args) =>
        Host.CreateDefaultBuilder(args)
            .ConfigureWebHostDefaults(webBuilder =>
            {
                webBuilder.UseStartup<Startup>();
            });
}

從上面的圖中可以看到,在我們的 Web 應用的宿主程式還未啟動之前,控制台就已經列印出了我們自己設定的信息,之後,才是啟動我們的 Web 應用,這裡是請求我們的 API 介面。同時可以發現,在模擬多次請求時,並不會再次觸發我們預設的事件。

 

三、總結

這一篇文章中並沒有包含代碼,更多的是針對我之前在開發中遇到的一個問題,自己在解決過程中的一個案例說明,希望可以在你以後遇到這類問題時可以提供一些幫助。離 2020 年的農曆新年也沒有幾天了,按目前的進度,估計就是年前的最後一篇博客了,我也要收拾收拾心情,準備過年了。最後,送大家一張表情包,獻給得知你是最後一個放假的童鞋,哈哈哈,提前祝大家新年快樂丫。

 

四、參考

  1. [ASP.NET Core 3框架揭秘] 依賴註入[8]:服務實例的生命周期
  2. 200行代碼,7個對象——讓你瞭解 ASP.NET Core 框架的本質
  3. ASP.NET Core 2.0 : 七.一張圖看透啟動背後的秘密
  4. ASP.NET Core 3.0 的新增功能

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本篇博客園是被任務所逼,而已有的使用nopi技術的文檔技術經驗又不支持我需要的應對各種複雜需求的苛刻要求,只能自己造輪子封裝了,由於需要應對很多總類型的數據採集需求,因此有了本篇博客的代碼封裝,下麵一點點介紹吧: 收集excel你有沒有遇到過一下痛點: 1-需要收集指定行標題位置的數據,我的標題行不 ...
  • linqtocsv文件有不太好的地方就是:無法設置標題的行數,預設首行就是標題,這不是很尷尬嗎? 並不是所有的csv文件嚴格寫的首行是標題,下麵全是數據,我接受的任務就是讀取很多.csv報表數據,裡面就有很多前幾行是說明性內容,下麵才是標題和數據。為了更好的解決這個問題,自己寫吧... 本博客沒有照 ...
  • 本筆記摘抄自:https://www.cnblogs.com/PatrickLiu/p/7723225.html,記錄一下學習過程以備後續查用。 一、引言 今天我們要講結構型設計模式的第三個模式--裝飾模式。當第一次看到這個名稱時想到的是另外一個詞語“裝修”,個人觀點談談對“裝修”的理解吧,請大家 ...
  • 在上一篇文章:《閃電光速拳? .NetCore 中的Span》中我們提到了在.net core 2.x 所新增的一個類型:Span。但是您會發現它無法用在我們項目的某些地方,它獨特的 ref結構 使它沒有辦法跨線程使用、更沒有辦法使用Lambda表達式。所以,這個時候如果我們又想跨線程操作數據又想獲... ...
  • 前言:想在.net framework環境使用自定義定時器的話,參考我的另一篇文章:https://www.cnblogs.com/lxhbky/p/10242839.html 想在.net core中使用定時器功能,需要藉助一個服務介面:IHostedService, 繼承並實現對應方法,最後再s ...
  • 要點 導出特性 如何導出Excel表頭 如何導出數據、如何進行數據的切割、如何使用篩選器 導出特性 ExporterAttribute + Name : 名稱(當前Sheet 名稱) + HeaderFontSize :頭部字體大小 + FontSize :正文字體大小 + MaxRowNumber ...
  • 你是否在初學 .net core時,被依賴註入所折磨? 你是否在開發過程中,為了註入依賴而不停的在Startup中增加註入代碼,而感到麻煩? 你是否考慮過或尋找過能輕鬆實現自動註入的組件? 如果有,那請歡迎繼續往下看。 或許你是被我這標題給吸引過來的,請不要懷疑自己的眼睛,如果你真的遇到過以上的問題 ...
  • 序言: 在 UWP 中,常見的存儲數據方式基本上就兩種。第一種方案是 UWP 框架提供的 ApplicationData Settings 這一系列的方法,適用於存放比較輕量的數據,例如存個 Boolean 類型的設置項這種是最適合不過的了。另一種方案是用 Sqlite 這種資料庫,適合存放數據量大 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...