ASP.NET Core - 依賴註入(三)

来源:https://www.cnblogs.com/wewant/archive/2023/02/28/17110701.html
-Advertisement-
Play Games

4. 容器中的服務創建與釋放 我們使用了 IoC 容器之後,服務實例的創建和銷毀的工作就交給了容器去處理,前面也講到了服務的生命周期,那三種生命周期中對象的創建和銷毀分別在什麼時候呢。以下麵的例子演示以下: 首先是新增三個類,用於註冊三種不同的生命周期: public class Service1 ...


4. 容器中的服務創建與釋放

我們使用了 IoC 容器之後,服務實例的創建和銷毀的工作就交給了容器去處理,前面也講到了服務的生命周期,那三種生命周期中對象的創建和銷毀分別在什麼時候呢。以下麵的例子演示以下:

首先是新增三個類,用於註冊三種不同的生命周期:

public class Service1
{
    public Service1()
    {
        Console.WriteLine("Service1 Created");
    }
}
public class Service2
{
    public Service2()
    {
        Console.WriteLine("Service2 Created");
    }
}
public class Service3
{
    public Service3()
    {
        Console.WriteLine("Service3 Created");
    }
}

接下來是演示場景,為了簡單起見,就用後臺服務程式吧

IHost host = Host.CreateDefaultBuilder(args)
    .ConfigureServices(services =>
    {
        services.AddHostedService<Worker>();
        services.AddSingleton<Service1>();
        services.AddScoped<Service2>();
        services.AddTransient<Service3>();
    })
    .Build();

await host.RunAsync();

public class Worker : BackgroundService
{
    private readonly ILogger<Worker> _logger;
    private readonly IServiceProvider _serviceProvid
    public Worker(ILogger<Worker> logger, IServiceProvider serviceProvider)
    {
        _logger = logger;
        _serviceProvider = serviceProvider;
    }

    protected override async Task ExecuteAsync(CancellationToken stoppingToken)
    {
        #region 生命周期實例創建
        Console.WriteLine("Service1 第一次調用");
        var service11 = _serviceProvider.GetService<Service1>();
        Console.WriteLine("Service1 第二次調用");
        var service12 = _serviceProvider.GetService<Service1>();

        // 創建作用域,與 Web 應用中的一次請求一樣
        using (var scope = _serviceProvider.CreateScope())
        {
            Console.WriteLine("Service2 第一次調用");
            var service31 = scope.ServiceProvider.GetService<Service2>();
            Console.WriteLine("Service2 第二次調用");
            var service32 = scope.ServiceProvider.GetService<Service2>();

            using (var scope1 = _serviceProvider.CreateScope())
            {
                Console.WriteLine("Service2 第三次調用");
                var service33 = scope1.ServiceProvider.GetService<Service2>();
            }
        }
        {
            Console.WriteLine("Service3 第一次調用");
            var service41 = _serviceProvider.GetService<Service3>();

            Console.WriteLine("Service3 第二次調用");
            var service42 = _serviceProvider.GetService<Service3>();
            }
            #endregion
        }
    }
}

最終的輸出如下:

image

通過輸出,我們可以單例生命周期服務在第一次使用的時候創建,之後一直是一個實例,作用域生命周期服務在一個作用域中第一次使用的時候創建實例,之後在同一個實例中只保持一個,但在其他作用域中則會重新創建,而瞬時生命周期服務每次都會創建一個新實例。

看完創建,我們再看實例銷毀的時機。

若服務實現了IDisposable介面,並且該服務是由DI容器創建的,則我們不應該手動去Dispose,DI容器會對服務自動進行釋放。這裡由兩個關鍵點,一個是要實現 Idisposable 介面,一個是由容器創建。這裡再增加多兩個類,用於演示,並且為了避免干擾將之前演示創建過程的代碼註釋。

public class Service1 : IDisposable
{
    public Service1()
    {
        Console.WriteLine("Service1 Created");
  
    public void Dispose()
    {
        Console.WriteLine("Service1 Dispose");
    }
}

public class Service2 : IDisposable
{
    public Service2()
    {
        Console.WriteLine("Service2 Created");

    public void Dispose()
    {
        Console.WriteLine("Service2 Dispose");
    }
}

public class Service3 : IDisposable
{
    public Service3()
    {
        Console.WriteLine("Service3 Created");
    }

    public void Dispose()
    {
        Console.WriteLine("Service3 Dispose");
    }
}

public class Service4 : IDisposable
{
    public void Dispose()
    {
        Console.WriteLine("Service4 Dispose");
    }
}

public class Service5 : IDisposable
{
    public void Dispose()
    {
        Console.WriteLine("Service5 Dispose");
    }
}

之後後臺服務程式也做一些修改

IHost host = Host.CreateDefaultBuilder(args)
    .ConfigureServices(services =>
    {
        services.AddHostedService<Worker>();
        services.AddSingleton<Service1>();
        services.AddScoped<Service2>();
        services.AddTransient<Service3>();
        // 這種方式依舊由容器創建實例,只不過提供了工廠方法
        services.AddSingleton<Service4>(provider => new Service4());
        // 這種方式是用外部創建實例,只有單例生命周期可用
        services.AddSingleton<Service5>(new Service5());
    })
    .Build();

await host.RunAsync();

public class Worker : BackgroundService
{
    private readonly ILogger<Worker> _logger;
    private readonly IServiceProvider _serviceProvid
    public Worker(ILogger<Worker> logger, IServiceProvider serviceProvider)
    {
        _logger = logger;
        _serviceProvider = serviceProvider;
    }

    protected override async Task ExecuteAsync(CancellationToken stoppingToken)
    {
        #region 生命周期實
        Console.WriteLine("Service1 調用");
        var service1 = _serviceProvider.GetService<Service1>();

        // 創建作用域,與 Web 應用中的一次請求一樣
        using (var scope = _serviceProvider.CreateScope())
        {
            Console.WriteLine("Service2 調用");
            var service2 = scope.ServiceProvider.GetService<Service2>();
            Console.WriteLine("即將結束作用域
            Console.WriteLine("Service3 調用");
            var service3 = scope.ServiceProvider.GetService<Service3>();
        }

        Console.WriteLine("Service4 調用");
        var service4 = _serviceProvider.GetService<Service4>();
        Console.WriteLine("Service5 調用");
        var service5 = _serviceProvider.GetService<Service5>();

        #endregion
    }
}

這樣要直接用命令啟動應用,不能夠通過vs調試,之後Ctrl+C停止應用的時候,輸出如下:

image

通過輸出可以看得到,瞬時生命周期服務和作用域生命周期服務在超出作用範圍就會被釋放,而單例生命周期服務則在應用關閉時才釋放,同為單例生命周期的Service5沒有被釋放。

這裡要提一下的是,在解析瞬時生命周期服務Service3的時候,示例代碼中是放到一個單獨的作用域中的,這是因為在通過 services.AddHostedService<Worker>(); 註入Worker的時候是註入為單例生命周期的,而在單例生命周期對象中解析其他生命周期的對象是會有問題的,這也是服務註入、解析需要註意的一個關鍵點。

一定要註意服務解析範圍,不要在 Singleton 中解析 Transient 或 Scoped 服務,這可能導致服務狀態錯誤(如導致服務實例生命周期提升為單例,因為單例生命周期的服務對象只會在應用停止的時候釋放,而單例對象都沒釋放,它的依賴項肯定不會釋放)。允許的方式有:

  • 在 Scoped 或 Transient 服務中解析 Singleton 服務
  • 在 Scoped 或 Transient 服務中解析 Scoped 服務(不能和前面的Scoped服務相同)

如果要在單例生命周期示例中臨時解析作用域、瞬時生命周期的服務,可以通過創建一個子作用域的方式。對子作用域 IServiceScope 的工作方式感興趣的,可閱讀一下這篇文章:細聊.Net Core中IServiceScope的工作方式



參考文章:
ASP.NET Core 依賴註入 | Microsoft Learn
理解ASP.NET Core - 依賴註入(Dependency Injection)



ASP.NET Core 系列:

目錄:ASP.NET Core 系列總結
上一篇:ASP.NET Core - 依賴註入(二)


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

-Advertisement-
Play Games
更多相關文章
  • 錯誤描述 在 Spring Cloud 項目中通過 Open Feign 遠程調用時出現如下錯誤: feign.codec.EncodeException: No qualifying bean of type 'org.springframework.boot.autoconfigure.http ...
  • 之前在做某個業務中,寫了個文件傳輸的程式,程式邏輯很簡單:掃描某個目錄下的文件,對文件進行一些處理,然後把文件移動到另一個目錄。 此前在大多數運行環境里,該程式一直正常運行,直到最近在一個新環境下,出現問題:文件移動失敗。查詢日誌發現在調用file.renameTo方法返回false。我第一反應是查 ...
  • C語言對記憶體的使用劃分為以下區域: 棧區(stack)、堆區(heap)、全局區(靜態區)、常量區、代碼區。 棧區: 由編譯器自動分配釋放,按記憶體地址從高(地址)到低(地址)存儲; 棧區內容的作用域為其所定義的函數內,生命周期為函數執行期間,函數結束自動釋放; 存放局部變數、const局部變數、函數 ...
  • 在上文中分析了 HttpURLConnection的用法,功能還是比較簡單的,沒有什麼封裝 接下來看看Apache HttpClient是如何封裝httpClient的 組成 HttpClient 5 的系統架構主要由以下幾個部分組成: HttpCore:核心包,包含了 HTTP 協議的核心抽象和實 ...
  • 背景 昨天,一位朋友找到我尋求幫助。他的項目需要調用一個第三方項目的webAPI。這個webAPI本身可從header, query string中取相關信息,但同事發現他在調用時,無法按期望的那樣從query string中傳參數給到第三方webAPI (webAPI仿佛忽略了從query str ...
  • 在預設情況下,WPF提供的DataGrid僅擁有數據展示等簡單功能,如果要實現像Excel一樣複雜的篩選過濾功能,則相對比較麻煩。本文以一個簡單的小例子,簡述如何通過WPF實話DataGrid的篩選功能,僅供學習分享使用,如有不足之處,還請指正。 ...
  • Avalonia 實現平滑拖動指定控制項 1.創建一個UserControl控制項,並且添加以下代碼 using System; using Avalonia; using Avalonia.Controls; using Avalonia.Input; using Avalonia.Markup.Xa ...
  • 1. 問題:安裝 Microsoft.Toolkit.Mvvm 運行後報錯:錯誤 CS0012 類型“Object”在未引用的程式集中定義。必須添加對程式集“netstandard, Version=2.0.0.0, Culture=neutral, PublicKeyToken=cc7b13ffc ...
一周排行
    -Advertisement-
    Play Games
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...