asp.net core 系列 4 註入服務的生存期

来源:https://www.cnblogs.com/MrHSR/archive/2019/01/08/10234436.html
-Advertisement-
Play Games

一.服務的生存期 在容器中每個註冊的服務,根據程式應用需求都可以選擇合適的服務生存期,ASP.NET Core 服務有三種生存期配置: (1) Transient:暫時生存期,在每次請求時被創建。 這種生存期適合輕量級的,無狀態的服務。 (2) Scoped: 作用域生存期,在每次請求被創建一次。 ...


一.服務的生存期

  在容器中每個註冊的服務,根據程式應用需求都可以選擇合適的服務生存期,ASP.NET Core 服務有三種生存期配置:

    (1) Transient:暫時生存期,在每次請求時被創建。 這種生存期適合輕量級的,無狀態的服務。

    (2) Scoped: 作用域生存期,在每次請求被創建一次。

    (3) Singleton: 單例生存期,在它們第一次被請求時創建。每個後續請求將使用相同的實例。如果應用程式需要單例行為,建議讓服務容器管理服務的生命周期,而不是在自己的類中實現單例模式。

   1.1 演示案例

    為了演示生存期和註冊選項之間的差異, 以下服務介面,任務是演示標識符 OperationId 的操作值變化。 根據為以下介面配置操作服務的生存期的方式,容器在類請求時提供相同或不同的服務實例:

   public interface IOperation
    {
        Guid OperationId { get; }
    }
    //用於演示暫時生存期
    public interface IOperationTransient : IOperation
    {
    }
    //用於演示作用域生存期
    public interface IOperationScoped : IOperation
    {
    }
    //用於演示單例生存期
    public interface IOperationSingleton : IOperation
    {
    }
    //用於演示單例中空GUID
    public interface IOperationSingletonInstance : IOperation
    {
    }

    上面四種服務介面在 Operation 類中實現。 調用 Operation類時將自動生成一個 GUID(如果實例化Operation類時沒有指定GUID),下麵是Operation類的實現:

 public class Operation : IOperationTransient,
     IOperationScoped,
     IOperationSingleton,
     IOperationSingletonInstance
    {
        /// <summary>
        /// 構造方法中生成GUID,在實例化類時
        /// </summary>
        public Operation() : this(Guid.NewGuid())
        {
        }

        public Operation(Guid id)
        {
            OperationId = id;
        }

        /// <summary>
        /// 獲取GUID
        /// </summary>
        public Guid OperationId { get; private set; }
    }

    再註冊一個 OperationService 服務,該服務取決於每個其他 Operation 類型。 當通過依賴關係註入請求 OperationService 時,它將接收每個服務的新實例或基於從屬服務(Operation )的生存期的現有實例。OperationService 服務作用就是第二次調用 Operation類,查看Operation類實例的作用域變化。

    public class OperationService
    {

        public IOperationTransient TransientOperation { get; }
        public IOperationScoped ScopedOperation { get; }
        public IOperationSingleton SingletonOperation { get; }
        public IOperationSingletonInstance SingletonInstanceOperation { get; }


        public OperationService(
        IOperationTransient transientOperation,
        IOperationScoped scopedOperation,
        IOperationSingleton singletonOperation,
        IOperationSingletonInstance instanceOperation)
        {
            TransientOperation = transientOperation;
            ScopedOperation = scopedOperation;
            SingletonOperation = singletonOperation;
            SingletonInstanceOperation = instanceOperation;
        }
    }

    (1) 如果在請求時創建了臨時服務(Transient),則 IOperationTransient 服務的 OperationId 與 OperationService 的 OperationId 不同。 OperationService 將接收 IOperationTransient 類的新實例。 新實例將生成一個不同的 OperationId。

    (2) 如果按請求創建有作用域的服務(Scoped),則 IOperationScoped 服務的 OperationId 與請求中 OperationService 的該 ID 相同。 在請求中,兩個服務共用不同的 OperationId 值。

    (3) 如果單一實例服務(Singleton),則只創建一次 併在所有請求和所有服務中使用,則 OperationId 在所有服務請求中保持不變。

 

    下麵是在 Startup.ConfigureServices 服務容器中註冊,指定服務的生存期:

            services.AddTransient<IOperationTransient, Operation>();
            services.AddScoped<IOperationScoped, Operation>();
            services.AddSingleton<IOperationSingleton, Operation>();
            services.AddSingleton<IOperationSingletonInstance>(new Operation(Guid.Empty));

            services.AddTransient<OperationService, OperationService>();

 

    為了演示各個請求中的對象生存期。 下麵示例應用 Index頁面,請求 IOperation 類型和 OperationService。 然後查看Operation類屬性OperationId 值的變化:

public class IndexModel : PageModel
    {
        public OperationService OperationService { get; }
        public IOperationTransient TransientOperation { get; }
        public IOperationScoped ScopedOperation { get; }
        public IOperationSingleton SingletonOperation { get; }
        public IOperationSingletonInstance SingletonInstanceOperation { get; }
   
        public IndexModel(
        OperationService operationService,
        IOperationTransient transientOperation,
        IOperationScoped scopedOperation,
        IOperationSingleton singletonOperation,
        IOperationSingletonInstance singletonInstanceOperation)
        {
            OperationService = operationService;
            TransientOperation = transientOperation;
            ScopedOperation = scopedOperation;
            SingletonOperation = singletonOperation;
            SingletonInstanceOperation = singletonInstanceOperation;
        }

        public string BindGUIDMsg { get; set; }
        public void OnGet()
        {
            BindGUIDMsg += "IOperation操作: <br/> ";
            BindGUIDMsg += "暫時性:" + TransientOperation.OperationId.ToString() + "</br>";
            BindGUIDMsg += "有作用域:" + ScopedOperation.OperationId.ToString() + "</br>";
            BindGUIDMsg += "單一實例:" + SingletonOperation.OperationId.ToString() + "</br>";
            BindGUIDMsg += "實例:" + SingletonInstanceOperation.OperationId.ToString() + "</br>";

            BindGUIDMsg += "</br></br></br>OperationService操作:</br>";
            BindGUIDMsg += "暫時性:" + OperationService.TransientOperation.OperationId.ToString() + "</br>";
            BindGUIDMsg += "有作用域:" + OperationService.ScopedOperation.OperationId.ToString() + "</br>";
            BindGUIDMsg += "單一實例:" + OperationService.SingletonOperation.OperationId.ToString() + "</br>";
            BindGUIDMsg += "實例:" + OperationService.SingletonInstanceOperation.OperationId.ToString() + "</br>";
        }
    }
    <div >
        @{
          @Html.Raw(@Model.BindGUIDMsg);  
        }
    </div>

 

    第一次Index頁面請求:

    IOperation 操作:
    暫時性:8ef874a3-743d-4288-98d4-3df126cd940d 
    有作用域:256ff050-f469-4ea3-8dde-16cdd3087c83 
    單一實例:d2caf297-a9b1-4dcf-ADDA-c68e46fe0741 
    實例:00000000-0000-0000-0000 -000000000000 

    OperationService操作:
    暫時性:5411fd0d-f2e1-4885-beee-2d7ccf48dceb 
    有作用域:256ff050-f469-4ea3-8dde-16cdd3087c83 
    單一實例:d2caf297-a9b1-4dcf-adda-c68e46fe0741 
    實例:00000000-0000-0000- 0000-000000000000

 

    第二次Index頁面請求:

    IOperation操作:
    暫時性:e685fd0e-d2e0-4900-9eff-e6bc41cd2f80
    有作用域:ca233b49-8326-4a7e-8ee4-6993d70786ed
    單一實例:d2caf297-a9b1-4dcf-adda-c68e46fe0741
    實例:00000000-0000-0000-0000-000000000000

    OperationService操作:
    暫時性:db89be00-c3b7-4f99-bead-5be693ccc2c0
    有作用域:ca233b49-8326-4a7e-8ee4-6993d70786ed
    單一實例:d2caf297-a9b1-4dcf-adda-c68e46fe0741
    實例:00000000-0000-0000-0000-000000000000

 

  下麵再總結一下:

    (1)暫時性註冊的服務,每次調用服務都會是一個新的服務對象實例。相當於在IndexModel類的局部(方法或屬性中)實例化一個依賴對象Operation類,偽代碼是:

public class IndexModel
{
    public void OnGet()
        {
//載入index頁時,實例化了二次Operation類
         //第一次 OperationService operationService
=new OperationService();
         //第二次
         IOperationTransient TransientOperation =new Operation(); } }

    (2)作用域註冊的服務,一次請求內(載入一次index頁)對象實例是相同的,但每次請求會產生一個新實例。相當於在IndexModel類的全局中實例化一次依賴對象Operation類,偽代碼是:

        OperationService operationService = null;
        public IndexModel()
        {
            operationService = new OperationService();
            operationService.ScopedOperation = new Operation();
        }

        public void OnGet()
        {
            operationService.ScopedOperation.OperationId;
            IOperationScoped operationScoped=operationService.ScopedOperation;
            operationScoped.OperationId
        }

    (3)單例註冊的服務,實例對象對每個對象和每個請求都是相同的。相當於在整個應用Application中只實例化一次,常見的單例模式。

 

  參考文獻:

    官方文檔:ASP.NET Core 

 


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

-Advertisement-
Play Games
更多相關文章
  • C#6.0的一些語法 一、Lambda表達式的一些運用 1.在屬性、方法中應用 例如常規對屬性的賦值: 使用Lambda表達式進行屬性賦值方法: public bool IsNagivation => this._navigationDestaion != null; 二、?和??的應用 1.?的應 ...
  • 到了ASP.NET Core,項目結構也發生了一些變化,看下麵的圖片。 共有6點跟.NET Framework不一樣,下麵來一一說明。1,lauchSettings.json該文件用於指定應用程式的啟動設置。2,wwwroot目錄一般用來放靜態資源文件,比如:js,css,img,ico等。3,依賴... ...
  • 最近頻繁使用非同步所以自己綜合的學習了一把非同步相關的知識,自己稍加整理了一下(這也是我試著寫的第一篇,如果有不對的,希望大神來指正!) 首先是 委托實現的非同步 class Program { public delegate int weituo();//定義了個委托 public int xxx() ...
  • 註冊按鈕事件: private void btnRegister_Click(object sender, EventArgs e) { string username = txtUserName.Text; string userpwd = txtUserPwd.Text; string tel ...
  • 上一篇我們講了關於direct類型的Exchange,這一片我們來瞭解一下fanout類型的Exchange。 1.Exchange的fanout類型 fanout類型的Exchange的特點是會把消息發送給與之綁定的所有Queue中,我們來測試一下。代碼如下 運行代碼,去可視化工具中查看一下 消費 ...
  • 主要功能: 所編寫的程式需將串口1、串口2數據互通,即:串口1接收到數據的同時將數據通過串口2發出,串口2接收到數據的同時將數據通過串口1發出。 並根據需要由指定串口發送或獲取數據。 代碼如下: ...
  • Acrobat.dllc#PDFPDFRender4NET.dllpdf轉圖片 GitHub Clone Adress : https://github.com/stone0090/OfficeTools.Pdf2Image.Word2Image.git (you get it). 前段時間公司安排 ...
  • 緣起 哈嘍大家周二好呀,剛剛經歷過了幾天火車搶票,整個人都不好了,不知道小伙伴對今年的春節是否還一如既往的期待呢,眼看都要春節了,本來也想寫篇2018總結篇,但是怕不免會出現雞湯文的窠臼嫌疑,想想還是算了,這幾天和老李聊起來關於寫博客文章,總感覺這一系列還沒有寫完,或者說還有一些沒有收尾好,眼看就要 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...