<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 可以保证在一个请求的处理过程中,所有的数据库操作都在同一个事务中进行,避免出现数据不一致的问题。

相关推荐
ghost14316 分钟前
Python自学第2天:条件语句,循环语句
开发语言·python·学习
Chandler2427 分钟前
Go:低级编程
开发语言·后端·golang
^_^ 纵歌41 分钟前
用python比较两个mp4是否实质相同
开发语言·python·音频·视频
一直走下去-明42 分钟前
使用python帮助艺术家完成角色动画和服装模型等任务
开发语言·图像处理·pytorch·python·opencv·ai作画
可观测性用观测云1 小时前
.NET Core 服务实现监控可观测性最佳实践
.net
长流小哥1 小时前
Linux网络编程实战:从字节序到UDP协议栈的深度解析与开发指南
linux·c语言·开发语言·网络·udp
幼儿园园霸柒柒1 小时前
第七章:7.2求方程a*x*x+b*x+c=0的根,用3个函数,分别求当:b*b-4*a*c大于0、等于0和小于0时的根并输出结果。从主函数输入a、b、c的值
c语言·开发语言·算法·c#
不知道叫什么呀1 小时前
【C语言基础】C++ 中的 `vector` 及其 C 语言实现详解
c语言·开发语言·c++
muyouking112 小时前
0.深入探秘 Rust Web 框架 Axum
开发语言·前端·rust
勇敢牛牛_2 小时前
【Rust基础】使用Rocket构建基于SSE的流式回复
开发语言·后端·rust