0

0

c# MediatR 在ASP.NET Core中是单例还是瞬时 线程安全吗

幻夢星雲

幻夢星雲

发布时间:2026-01-07 08:23:30

|

710人浏览过

|

来源于php中文网

原创

MediatR 的 IMediator 默认注册为瞬时(Transient),每次解析都创建新实例;其线程安全性取决于 handler 的实现与生命周期配置,错误地将 Scoped 服务注入 Singleton handler 会导致运行时异常。

c# mediatr 在asp.net core中是单例还是瞬时 线程安全吗

MediatR 默认注册为瞬时(Transient)

在 ASP.NET Core 中,MediatRIMediator 接口默认通过 AddMediatR() 扩展方法注册为 ServiceLifetime.Transient。这意味着每次从 DI 容器解析 IMediator 时,都会创建一个新实例。

它本身不持有跨请求的共享状态,但它的行为依赖于你注册的 handlers 和 pipeline behaviors:

  • INotificationHandlerIRequestHandler 等 handler 类型,默认也按 Transient 注册(除非你显式改用 ScopedSingleton
  • 如果你把某个 handler 注册为 Singleton,而它内部又持有非线程安全的状态(比如普通字段、静态集合),那就可能出问题
  • MediatR 内部使用 IServiceProvider 解析 handler,所以 handler 的生命周期必须 ≥ IMediator 实例的生命周期(否则会抛 InvalidOperationException

MediatR 是线程安全的,但 handler 不一定

MediatR 核心类型(如 Mediator 类)本身是无状态的,所有操作都委托给 DI 解析出的 handler,因此 IMediator 实例可被多线程并发调用 —— 这是安全的。

真正决定线程安全的是你写的 handler:

  • 如果 handler 是纯函数式(只读参数、不改内部字段、不操作静态变量),那它是线程安全的
  • 如果 handler 里用了 static List Cache = new() 并直接 Add,就会发生竞态
  • 如果 handler 依赖 DbContext(Scoped),而你把它错误注册为 Singleton,运行时可能抛 InvalidOperationException: A second operation started on this context before a previous operation completed

常见错误:在 Singleton handler 中注入 Scoped 服务

这是最典型的生命周期冲突,会导致运行时异常或数据错乱。例如:

public class MyHandler : IRequestHandler
{
    private readonly ApplicationDbContext _db;

    public MyHandler(ApplicationDbContext db) // ← DbContext 是 Scoped
    {
        _db = db;
    }

    public Task Handle(MyQuery request, CancellationToken ct)
    {
        return _db.Users.CountAsync(ct);
    }
}

如果你这样注册:

Cursor Directory
Cursor Directory

专为Cursor设计的开源资源库、提示词库

下载
services.AddSingleton, MyHandler>(); // ❌ 错误

就会导致同一个 MyHandler 实例被多个请求复用,而它持有的 _db 是 Scoped,早已被释放或正在被其他线程使用。

正确做法是让 MediatR 自动发现并注册为 Transient:

services.AddMediatR(cfg => cfg.RegisterServicesFromAssembly(typeof(MyHandler).Assembly)); // ✅ 默认就是 Transient

需要单例行为?别改 MediatR,改你的设计

有时候你会想“让某个 handler 全局只初始化一次”,比如加载配置、连接第三方 SDK。这时不要把 handler 设为 Singleton,而是:

  • 把需要共享的状态提取到单独的 Singleton service 中(如 ISmsClientPool
  • 在 handler 中注入该 Singleton 服务
  • handler 本身仍保持 Transient,避免生命周期污染

这样既满足复用需求,又不破坏 DI 安全边界。MediatR 的设计哲学就是“消息即操作”,不是“消息即状态容器”。

真正容易被忽略的是:handler 的构造函数执行时机和作用域,远比 IMediator 本身的生命周期更关键。别盯着 MediatR 是单例还是瞬时,先盯紧你注册的每一个 handler 类型及其依赖树。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

314

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

529

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

49

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

193

2025.08.29

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1005

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

56

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

332

2025.12.29

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

476

2023.08.10

PPT动态图表制作教程大全
PPT动态图表制作教程大全

本专题整合了PPT动态图表制作相关教程,阅读专题下面的文章了解更多详细内容。

13

2026.01.07

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Java 教程
Java 教程

共578课时 | 43.3万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号