.NET8 依赖注入组件

前言: 在现代的软件开发中,依赖注入(Dependency Injection,简称DI)是一种重要的设计模式,它能够帮助我们更好地管理对象的依赖关系,提高代码的可维护性和可测试性。.NET框架提供了强大的依赖注入容器,可以方便地在应用程序中进行服务的注册和解析。本文将讲解如何使用微软官方.NET 8依赖注入组件 Microsoft.Extensions.DependencyInjection,以及几种常见的注入方式及其适用场景。

一、安装Microsoft.Extensions.DependencyInjection

使用 NuGet包下载安装组件,需要注意与所选择的.net环境一致(我这里创建的是.net8)

二、几种常见的注入方式及其适用场景

1、创建依赖注入容器

首先,我们需要创建一个ServiceCollection对象,该对象就是我们的依赖注入容器,用于注册服务。

csharp 复制代码
ServiceCollection services = new ServiceCollection();
2、注册服务

在.NET中,我们可以通过不同的生命周期来注册服务。主要有三种生命周期:单例(Singleton)瞬态(Transient)作用域(Scoped)。下面分别对这三种生命周期进行讲解。

2.1 单例(Singleton)

单例模式意味着在整个应用程序的生命周期内,服务只会被实例化一次。每次从依赖注入容器中解析该服务时,都会得到同一个对象实例。

csharp 复制代码
services.AddSingleton<IProductRepository, ProductRepository>();

业务场景:

①配置文件读取器:因为配置文件的内容在应用程序运行期间通常是不变的,所以不需要每次都重新读取。
②缓存:缓存类通常也需要在应用程序的整个生命周期内保持单例,以便所有组件都能共享同一个缓存实例。

2.2 瞬态(Transient)

瞬态模式意味着每次从依赖注入容器中解析服务时,都会创建一个新的服务实例。

csharp 复制代码
services.AddTransient<IProductRepository, ProductRepository>();

业务场景:

①日志记录器:在某些情况下,我们可能希望每个请求或操作都有独立的日志记录实例,以便单独记录每个请求或操作的日志。
②每次请求都需要独立状态的服务:例如,某些服务可能需要处理特定请求的状态,每次请求都需要一个新的实例来保持请求的独立性。

2.3 作用域(Scoped)

作用域模式意味着服务在每个请求或作用域内只会被实例化一次。如果同一个请求内解析多次,会得到同一个对象实例;如果不同请求内解析,会得到不同的对象实例。

csharp 复制代码
services.AddScoped<IProductRepository, ProductRepository>();

业务场景:

①数据库上下文:在Web应用程序中,通常希望每个请求都有自己独立的数据库上下文实例,以确保线程安全。
②会话数据处理:在某些情况下,我们需要处理每个会话的数据,作用域模式可以确保会话数据在同一次请求内的一致性。

3、解析服务

注册完服务后,我们需要构建服务提供者(IServiceProvider)来解析服务。服务提供者会根据我们注册的生命周期来创建服务实例。

csharp 复制代码
var serviceProvider = services.BuildServiceProvider();

然后,我们可以通过服务提供者来获取服务实例并调用其方法。

csharp 复制代码
IProductRepository productRepository = serviceProvider.GetRequiredService<IProductRepository>();
productRepository.GetProduct();
4、多实现类注册

在某些情况下,一个服务接口可能有多个实现类。我们可以注册这些实现类,并通过GetServices方法获取所有实现类的实例。

csharp 复制代码
services.AddScoped<IProductService, ProductService>();
services.AddScoped<IProductService, ProductService2>();

解析并调用这些实现类的方法:

csharp 复制代码
using (IServiceScope scope = serviceProvider.CreateScope())
{
    IEnumerable<IProductService> productServices = scope.ServiceProvider.GetServices<IProductService>();
    foreach (var productService in productServices)
    {
        productService.SaveProduct();
    }
}

业务场景:

①多种数据源:例如,一个应用程序需要从不同的数据库或API获取数据,我们可以为每种数据源创建一个服务实现类。
②插件系统:在支持插件的系统中,不同的插件可能实现同一个接口,我们可以将它们注册为不同的服务实现类,然后在运行时调用所有插件的方法。

三、三种方式的IOC依赖注入代码示例

csharp 复制代码
using Dunk.Common.Project.Injection.Injection;
using Microsoft.Extensions.DependencyInjection;

namespace Dunk.Common.Project.Injection
{
    internal class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine("Hello, World!");

            #region 使用 AddSingleton 注册
            //【1】 创建依赖注入(IOC容器)
            ServiceCollection services = new ServiceCollection();

            //【2】 注册服务,使用AddSingleton注册一个单例服务
            services.AddSingleton<IProductRepository, ProductRepository>(); // 注册另一个实现类
            services.AddSingleton<ProductService>(); // 注册具体类型


            //【3】 使用BuildServiceProvider取出服务
            var serviceProvider = services.BuildServiceProvider();

            //【4】 取对象,调用方法
            // 1、通过接口获取服务
            IProductRepository productRepository = serviceProvider.GetRequiredService<IProductRepository>();
            productRepository.GetProduct();

            // 2、或者通过具体类型获取服务
            ProductService productService = serviceProvider.GetRequiredService<ProductService>();
            productService.GetProduct();

            #endregion

            #region 使用 AddTransient 注册
            //【1】 创建依赖注入(IOC容器)
            ServiceCollection services = new ServiceCollection();

            //【2】 注册服务,使用AddSingleton注册一个单例服务
            services.AddTransient<IProductRepository, ProductRepository>(); // 注册另一个实现类
            services.AddTransient<ProductService>(); // 注册具体类型


            //【3】 使用BuildServiceProvider取出服务
            var serviceProvider = services.BuildServiceProvider();

            //【4】 取对象,调用方法
            // 1、通过接口获取服务
            IProductRepository productRepository = serviceProvider.GetRequiredService<IProductRepository>();
            productRepository.GetProduct();

            // 2、或者通过具体类型获取服务
            ProductService productService = serviceProvider.GetRequiredService<ProductService>();
            productService.GetProduct();

            #endregion


            #region 使用AddScoped 注册
            //【1】 创建依赖注入(IOC容器)
            ServiceCollection services = new ServiceCollection();

            //【2】 注册服务,使用AddSingleton注册一个单例服务
            services.AddScoped<IProductRepository, ProductRepository>(); // 注册另一个实现类
            services.AddScoped<ProductService>(); // 注册具体类型


            //【3】 使用BuildServiceProvider取出服务
            var serviceProvider = services.BuildServiceProvider();

            //【4】 取对象,调用方法

            using (IServiceScope scope = serviceProvider.CreateScope())// 创建作用域,范围
            {
                IProductRepository productRepository = scope.ServiceProvider.GetRequiredService<IProductRepository>();//通过接口获取服务
                productRepository.GetProduct();

                ProductService productService = scope.ServiceProvider.GetRequiredService<ProductService>(); //通过具体类型获取服务
                productService.GetProduct();
            }

            using (IServiceScope scope = serviceProvider.CreateScope())// 创建作用域,范围
            {
                IProductRepository productRepository = scope.ServiceProvider.GetRequiredService<IProductRepository>();//通过接口获取服务
                productRepository.GetProduct();

                ProductService productService = scope.ServiceProvider.GetRequiredService<ProductService>(); //通过具体类型获取服务
                productService.GetProduct();
            }


            #endregion

            #region 使用多实现类注册
            //【1】 创建依赖注入(IOC容器)
            ServiceCollection services = new ServiceCollection();

            //【2】 注册服务,使用AddSingleton注册一个单例服务
            services.AddScoped<IProductRepository, ProductRepository>(); // 注册另一个实现类
            services.AddScoped<IProductService, ProductService>(); // 注册另一个实现类
            services.AddScoped<IProductService, ProductService2>(); // 注册另一个实现类
            //services.AddScoped<ProductService>(); // 注册具体类型
            //services.AddScoped<ProductService2>(); // 注册具体类型


            //【3】 使用BuildServiceProvider取出服务
            var serviceProvider = services.BuildServiceProvider();

            //【4】 取对象,调用方法

            using (IServiceScope scope = serviceProvider.CreateScope())// 创建作用域,范围
            {

                // 使用GetServices方法取多个实现类
                IEnumerable<IProductService> productServices = scope.ServiceProvider.GetServices<IProductService>();
                foreach (var productService in productServices)
                {
                    productService.SaveProduct();
                }

               
            }


            #endregion
        }
    }
}

四、总结

通过本文的讲解,了解了如何在.NET 8中使用依赖注入组件来注册和解析服务。同时,探讨了几种不同的生命周期模式及其适用的业务场景。

相关推荐
Java手札2 分钟前
Windows下Golang与Nuxt项目宝塔部署指南
开发语言·windows·golang
小生凡一4 分钟前
腾讯二面:TCC分布式事务 | 图解TCC|用Go语言实现一个TCC
开发语言·分布式·golang
minji...8 分钟前
C语言 函数递归
c语言·开发语言·算法
云上空28 分钟前
C#初级知识总结
开发语言·c#
钢铁男儿1 小时前
C# 深入理解类:面向对象编程的核心数据结构
开发语言·数据结构·c#
Doker 多克1 小时前
Python-Django系列—部件
开发语言·python
江沉晚呤时1 小时前
深入解析 ASP.NET Core 中的 ResourceFilter
开发语言·c#·.net·lucene
huangyuchi.2 小时前
【C++11】Lambda表达式
开发语言·c++·笔记·c++11·lambda·lambda表达式·捕捉列表
XiaoyuEr_66882 小时前
如何创建一个C#项目(基于VS2022版)
开发语言·c#
Mercury-circle2 小时前
JavaScript基础知识合集笔记1——数据类型
开发语言·javascript·笔记