.net core mvc啟動順序以及主要部件3-Startup

来源:https://www.cnblogs.com/lvshunbin/archive/2019/06/30/11110615.html
-Advertisement-
Play Games

前面分享了.net core Program類的啟動過程已經源代碼介紹,這裡將繼續講Startup類中的兩個約定方法,一個是ConfigureServices,這個方法是用來寫我們應用程式所依賴的組件。另一個Configure,它是我們MVC請求的中間件方法,也就是我們每個請求來要執行的過程都可以寫 ...


前面分享了.net core Program類的啟動過程已經源代碼介紹,這裡將繼續講Startup類中的兩個約定方法,一個是ConfigureServices,這個方法是用來寫我們應用程式所依賴的組件。另一個Configure,它是我們MVC請求的中間件方法,也就是我們每個請求來要執行的過程都可以寫在這個方法裡面。
      為什麼說Startup類中的兩個方法是基於約定的呢?其實是這樣的,在.net core Program類Main方法中有個調用了Run方法這個方法從IServiceCollection容器中拿到一個IStartup類型的實例然後調用了IStartup中定義的兩個方法法,如果我們的Startup類是實現了這個介面的類 那麼就不是基於約定了,直接就可以使用,但是我們發現在vs給我們生成的Startup類並沒有實現任何介面,所以就不會是IStartup類型,那麼內部是如何去做的呢?  其實是這樣的,在註冊Startup實例的時候還有個類型叫做ConventionBasedStartup從名稱上解讀這個類就是轉換為基礎的Startup,其實卻是也是這樣的,這個類中是實現了IStartup介面,它的兩個方法中分別調用了各自的對用委托,這些委托實際執行的就是我們Startup類中定義的兩個方法,請看源代碼:

 public class ConventionBasedStartup : IStartup
    {
        private readonly StartupMethods _methods;

        public ConventionBasedStartup(StartupMethods methods)
        {
            _methods = methods;
        }

        public void Configure(IApplicationBuilder app)
        {
            try
            {
                _methods.ConfigureDelegate(app);
            }
            catch (Exception ex)
            {
                if (ex is TargetInvocationException)
                {
                    ExceptionDispatchInfo.Capture(ex.InnerException).Throw();
                }
                throw;
            }
        }

        public IServiceProvider ConfigureServices(IServiceCollection services)
        {
            try
            {
                return _methods.ConfigureServicesDelegate(services);
            }
            catch (Exception ex)
            {
                if (ex is TargetInvocationException)
                {
                    ExceptionDispatchInfo.Capture(ex.InnerException).Throw();
                }
                throw;
            }
        }
    }

  現在Startup類的方法說清楚了,我們具體來說說方法中的內容,首先說ConfigureServices(IServiceCollection services),這個方法的參數是約定好的,不能隨意改變,裡面的IServiceCollection介面其實就是我們依賴註入的容器,說的再直白一點就是我們整個MVC所需要的實例都由IServiceCollection所管理,IServiceCollection有幾個重要的擴展方法,他們都是定義在ServiceCollectionServiceExtensions靜態類中,AddTransient方法,表示用這個方法添加到IServiceCollection容器的實例在需要註入的實例中都是一個全新的實例,AddScoped方法,這個方法表示在一次請求的生命周期內共用一個實例,AddSingleton方法,這個方法表示整個程式共用一個實例,例如日誌服務,IConfiguration服務等都屬於典型Singleton。請看例子:

 public class Startup
    {
        public void ConfigureServices(IServiceCollection services)
        {
            services.AddTransient<TransientService>();
            services.AddTransient<ServiceDemo1>();
            services.AddTransient<ServiceDemo2>();
        }

        public void Configure(IApplicationBuilder app, ServiceDemo1 demo1, ServiceDemo2 demo2 )
        {
            demo1.Test();
            demo2.Test();
            app.Run(async (HttpContext context) =>
            {
              await  context.Response.WriteAsync("test successd");
            });
        }
    }
    public class TransientService
    {
        private int _updateCount;
        public int GetUpdateCount()
        {   
this._updateCount = this._updateCount + 1; return this._updateCount; } } public class ServiceDemo1 { private readonly TransientService _service; public ServiceDemo1(TransientService service) { _service = service; } public void Test() { Console.WriteLine($"我是demo1的計數:{this._service.GetUpdateCount()}"); } } public class ServiceDemo2 { private readonly TransientService _service; public ServiceDemo2(TransientService service) { _service = service; } public void Test() { Console.WriteLine($"我是demo2的計數:{this._service.GetUpdateCount()}"); } }

  上面的例子中會產生一下結果,可以看得出來這兩個註入的TransientService都是全新的實例

如果我們稍微改變一下註入的方法,將原本的 services.AddTransient<TransientService>();改成services.AddScoped<TransientService>();就會產生如下結果:

這個能說明什麼呢,我們有兩次註入  這個就表示TransientService保持了之前demo1的狀態  demo1和demo2是可以共用這個實例來傳輸數據的,AddSingleton方法理解起來比較簡單就不過多絮叨了,上面已經說明。

接下來再來說說Startup類中的Configure方法,Configure方法中的參數是可以變化的,也就是說你可以用依賴註入的方法在參數中註入你想要註入的實例,前面說了 這個方法是我們請求的中間件方法,這個方法中會整合我們註入的IApplicationBuilder 中調用的各種Use方法中定義的中間件  並不是說這裡面定義的代碼每次請求都會被執行,這個概念一定要搞清楚,Configure方法只會在啟動的時候執行一次,後面就不會再執行了,Configure方法只是讓我們可以定義整個MVC的處理請求的執行順序,具體的可以看看官方的文檔https://docs.microsoft.com/zh-cn/aspnet/core/fundamentals/middleware/?view=aspnetcore-2.2

其實中間件都是由IApplicationBuilder 所管理的ApplicationBuilder類實現了IApplicationBuilder 介面中的方法,看到ApplicationBuilder中的源代碼中有個屬性_components 它是一個IList<Func<RequestDelegate, RequestDelegate>>類型的委托容器,容器中的委托就是請求過來需要執行的中間件委托,當你在Configure方法中調用app.UseXXX的時候就會被註冊到這個容器中去,然後請求過來就按照順序執行容器中的每一個委托,所以這裡就解釋了前面說的Configure方法只會被執行一次的說法。下麵也貼一下ApplicationBuilder類的源代碼:

 public class ApplicationBuilder : IApplicationBuilder
    {
        private readonly IList<Func<RequestDelegate, RequestDelegate>> _components = new List<Func<RequestDelegate, RequestDelegate>>();

        public IServiceProvider ApplicationServices
        {
            get
            {
                return GetProperty<IServiceProvider>(Constants.BuilderProperties.ApplicationServices);
            }
            set
            {
                SetProperty(Constants.BuilderProperties.ApplicationServices, value);
            }
        }

        public IFeatureCollection ServerFeatures => GetProperty<IFeatureCollection>(Constants.BuilderProperties.ServerFeatures);

        public IDictionary<string, object> Properties
        {
            get;
        }

        public ApplicationBuilder(IServiceProvider serviceProvider)
        {
            Properties = new Dictionary<string, object>(StringComparer.Ordinal);
            ApplicationServices = serviceProvider;
        }

        public ApplicationBuilder(IServiceProvider serviceProvider, object server)
            : this(serviceProvider)
        {
            SetProperty(Constants.BuilderProperties.ServerFeatures, server);
        }

        private ApplicationBuilder(ApplicationBuilder builder)
        {
            Properties = new CopyOnWriteDictionary<string, object>(builder.Properties, StringComparer.Ordinal);
        }

        private T GetProperty<T>(string key)
        {
            if (!Properties.TryGetValue(key, out object value))
            {
                return default(T);
            }
            return (T)value;
        }

        private void SetProperty<T>(string key, T value)
        {
            Properties[key] = value;
        }

        public IApplicationBuilder Use(Func<RequestDelegate, RequestDelegate> middleware)
        {
            _components.Add(middleware);
            return this;
        }

        public IApplicationBuilder New()
        {
            return new ApplicationBuilder(this);
        }

        public RequestDelegate Build()
        {
            RequestDelegate requestDelegate = delegate (HttpContext context)
            {
                context.Response.StatusCode = 404;
                return Task.CompletedTask;
            };
            foreach (Func<RequestDelegate, RequestDelegate> item in _components.Reverse())
            {
                requestDelegate = item(requestDelegate);
            }
            return requestDelegate;
        }
    }

好啦,這篇關於Startup類就算介紹完成了,下篇開始正式介紹MVC





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

-Advertisement-
Play Games
更多相關文章
  • django 中間件 [TOC] 自定義中間件 中間件可以定義五個方法,分別是:(主要的是process_request和process_response) process_request(self,request) process_view(self, request, view_func, vi ...
  • MyBatis 類型處理器(typeHandlers)簡介及源碼分析 ...
  • 新聞 "逐漸演化的.NET Core框架" "Visual Studio提示與技巧" "Windows Termina(預覽)" "Microsoft在GitHub上的工程師從2000名增加至25000名" 視頻及幻燈片 "實用IRL" "Visual Studio的F 工具解析" "事件溯源DIY ...
  • 1)ThreadLocal如何回收value,什麼時候回收?從ThreadLocal中的內部類分析:① ThreadLocalMap是一個定製的哈希映射,僅適用於維護線程本地值。為了幫助處理非常大和長期使用的用法,哈希表條目使用weakreferences作為鍵。但是,由於不使用引用隊列,因此只有當 ...
  • 一、註解(Annotation) 1、什麼是註解? 從JDK5開始,Java增加了Annotation(註解),Annotation是代碼里的特殊標記,這些標記可以在編譯、類載入、運行時被讀取,並執行相應的處理。 2、Annotation與註釋的區別: (1)Annotation不是程式本身,可以對 ...
  • 上篇文章《 "在.NET Core 3.0中的WPF中使用IOC圖文教程" 》中,我們嘗試在WPF中應用 內置的IOC進行編程,在解析 的時候我用了 方法,當時就在想這個 方法跟 到底有什麼區別呢,於是乎,谷歌了一把,就發現了一篇文章來介紹他們區別的,於是乎嘗試翻譯一把,希望對大家有所幫助。文章最後 ...
  • 一、可空類型(Nullable Type)表示在值類型的正常取值範圍內再加上一個null值,聲明一個可空類型的標準語句如下: 例如,聲明一個可空整數類型並賦值為null: 1.可空類型是泛型結構Nullable<T>的實例,其聲明為:public struct Nullable<T> where T ...
  • C# -- 使用ODBC連接資料庫 ODBC DSN配置: ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...