<C#> 详细介绍.net 三种依赖注入:AddTransient、AddScoped、AddSingleton 的区别

在 .NET 8 里,AddTransientAddScopedAddSingleton 均为依赖注入容器用于注册服务的方法,不过它们的生命周期管理方式存在差异。下面为你详细介绍这三种方法的区别。

1. AddTransient

AddTransient 方法所注册的服务,每次被请求时都会创建一个新的实例。也就是说,在每次向依赖注入容器请求该服务时,容器都会生成一个全新的对象。

这种生命周期适用于那些无状态的服务,即服务的行为不会受到状态的影响,并且在每次使用时不需要保存任何状态信息。例如,数据验证服务、简单的计算服务等。

示例代码如下:

csharp 复制代码
using Microsoft.Extensions.DependencyInjection;

// 创建服务集合
var services = new ServiceCollection();

// 注册瞬态服务
services.AddTransient<ITransientService, TransientService>();

// 构建服务提供程序
var serviceProvider = services.BuildServiceProvider();

// 第一次请求服务
var transientService1 = serviceProvider.GetService<ITransientService>();
// 第二次请求服务
var transientService2 = serviceProvider.GetService<ITransientService>();

// 判断两个实例是否相同
bool isSameInstance = ReferenceEquals(transientService1, transientService2);
Console.WriteLine($"两个瞬态服务实例是否相同: {isSameInstance}"); // 输出: false

// 定义服务接口
public interface ITransientService
{
    void DoSomething();
}

// 实现服务接口
public class TransientService : ITransientService
{
    public void DoSomething()
    {
        Console.WriteLine("瞬态服务正在执行操作");
    }
}

2. AddScoped

AddScoped 方法注册的服务,在同一个服务作用域内只会创建一个实例。在 ASP.NET Core 应用程序里,一个 HTTP 请求通常会创建一个新的服务作用域,因此在同一个请求的处理过程中,所有对该服务的请求都会返回同一个实例。

这种生命周期适用于那些在一个请求处理过程中需要保持状态一致性的服务,比如数据库上下文服务。

示例代码如下:

csharp 复制代码
using Microsoft.Extensions.DependencyInjection;

// 创建服务集合
var services = new ServiceCollection();

// 注册作用域服务
services.AddScoped<IScopedService, ScopedService>();

// 构建服务提供程序
var serviceProvider = services.BuildServiceProvider();

// 创建一个新的服务作用域
using (var scope = serviceProvider.CreateScope())
{
    var scopedServiceProvider = scope.ServiceProvider;
    // 第一次请求服务
    var scopedService1 = scopedServiceProvider.GetService<IScopedService>();
    // 第二次请求服务
    var scopedService2 = scopedServiceProvider.GetService<IScopedService>();

    // 判断两个实例是否相同
    bool isSameInstance = ReferenceEquals(scopedService1, scopedService2);
    Console.WriteLine($"两个作用域服务实例是否相同: {isSameInstance}"); // 输出: true
}

// 定义服务接口
public interface IScopedService
{
    void DoSomething();
}

// 实现服务接口
public class ScopedService : IScopedService
{
    public void DoSomething()
    {
        Console.WriteLine("作用域服务正在执行操作");
    }
}

3. AddSingleton

AddSingleton 方法注册的服务,在整个应用程序的生命周期内只会创建一个实例。也就是说,无论何时向依赖注入容器请求该服务,都会返回同一个对象实例。

这种生命周期适用于那些无状态且需要全局共享的服务,比如配置服务、缓存服务等。

示例代码如下:

csharp 复制代码
using Microsoft.Extensions.DependencyInjection;

// 创建服务集合
var services = new ServiceCollection();

// 注册单例服务
services.AddSingleton<ISingletonService, SingletonService>();

// 构建服务提供程序
var serviceProvider = services.BuildServiceProvider();

// 第一次请求服务
var singletonService1 = serviceProvider.GetService<ISingletonService>();
// 第二次请求服务
var singletonService2 = serviceProvider.GetService<ISingletonService>();

// 判断两个实例是否相同
bool isSameInstance = ReferenceEquals(singletonService1, singletonService2);
Console.WriteLine($"两个单例服务实例是否相同: {isSameInstance}"); // 输出: true

// 定义服务接口
public interface ISingletonService
{
    void DoSomething();
}

// 实现服务接口
public class SingletonService : ISingletonService
{
    public void DoSomething()
    {
        Console.WriteLine("单例服务正在执行操作");
    }
}

总结

  • AddTransient:每次请求都会创建新实例,适用于无状态服务。
  • AddScoped:在同一个服务作用域内返回相同实例,适用于在请求处理过程中需保持状态一致的服务。
  • AddSingleton:在整个应用程序生命周期内只创建一个实例,适用于无状态且需全局共享的服务。

在大多数情况下,数据库操作服务适合使用 AddScoped 进行注册,这在 ASP.NET Core 应用中尤为常见。在这类应用里,每个 HTTP 请求都会创建一个新的服务作用域,在这个作用域内使用相同的数据库上下文实例能保证数据操作的一致性和事务性。
适用原因
状态一致性:同一个请求内,使用相同的数据库上下文实例可以保证数据状态的一致性。比如在处理一个业务逻辑时,可能会涉及多次数据库查询和更新操作,使用同一个上下文实例可以确保这些操作基于相同的数据快照。
事务处理:数据库事务通常需要在同一个上下文实例中完成。使用 AddScoped 可以保证在一个请求的处理过程中,所有的数据库操作都在同一个事务中进行,避免出现数据不一致的问题。

相关推荐
lllsure3 小时前
Python基础语法
开发语言·python
界面开发小八哥3 小时前
界面控件DevExpress WinForms v25.1新功能预览 - 功能区组件全新升级
人工智能·.net·界面控件·winform·devexpress
Kookoos4 小时前
ABP vNext + EF Core 实战性能调优指南
数据库·后端·c#·.net·.netcore
zxctsclrjjjcph4 小时前
【高并发内存池】从零到一的项目之centralcache整体结构设计及核心实现
开发语言·数据结构·c++·链表
zm4 小时前
服务器多客户端连接核心要点(1)
java·开发语言
炯哈哈4 小时前
【上位机——MFC】单文档和多文档视图架构
开发语言·c++·mfc·上位机
FuckPatience4 小时前
关于C#项目中 服务层使用接口的问题
java·开发语言·c#
CodeCraft Studio5 小时前
国产Excel处理控件Spire.XLS系列教程:C# 将Excel文件转换为Markdown格式
c#·excel
编程轨迹_5 小时前
使用 Spring 和 Redis 创建处理敏感数据的服务
java·开发语言·restful
赵和范5 小时前
C++:书架
开发语言·c++·算法