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

相关推荐
阿珊和她的猫3 小时前
v-scale-scree: 根据屏幕尺寸缩放内容
开发语言·前端·javascript
fouryears_234176 小时前
Flutter InheritedWidget 详解:从生命周期到数据流动的完整解析
开发语言·flutter·客户端·dart
我好喜欢你~6 小时前
C#---StopWatch类
开发语言·c#
lifallen7 小时前
Java Stream sort算子实现:SortedOps
java·开发语言
IT毕设实战小研7 小时前
基于Spring Boot 4s店车辆管理系统 租车管理系统 停车位管理系统 智慧车辆管理系统
java·开发语言·spring boot·后端·spring·毕业设计·课程设计
cui__OaO9 小时前
Linux软件编程--线程
linux·开发语言·线程·互斥锁·死锁·信号量·嵌入式学习
鱼鱼说测试9 小时前
Jenkins+Python自动化持续集成详细教程
开发语言·servlet·php
艾莉丝努力练剑10 小时前
【洛谷刷题】用C语言和C++做一些入门题,练习洛谷IDE模式:分支机构(一)
c语言·开发语言·数据结构·c++·学习·算法
一阵没来由的风10 小时前
拒绝造轮子(C#篇)ZLG CAN卡驱动封装应用
c#·can·封装·zlg·基础封装·轮子
CHEN5_0210 小时前
【Java基础面试题】Java基础概念
java·开发语言