0

0

C# 手动实现CQRS模式方法 C#如何不依赖MediatR实现CQRS

畫卷琴夢

畫卷琴夢

发布时间:2026-02-02 11:42:08

|

614人浏览过

|

来源于php中文网

原创

CQRS在C#中最简手动实现是通过分离ICommand/IQuery接口及对应处理器,命令只改状态无返回,查询只读数据不修改,类型不可变且职责明确。

c# 手动实现cqrs模式方法 c#如何不依赖mediatr实现cqrs

什么是CQRS在C#里最简可行的手动实现

CQRS(Command Query Responsibility Segregation)本质是把“改数据”和“读数据”彻底拆开——不是靠框架,而是靠接口分离、类型隔离和调用路径隔离。不依赖 MediatR 时,核心就是自己定义 ICommand / IQuery 接口,再配两个独立的处理器抽象,不共享输入/输出模型,也不共用执行管道。

手动定义命令与查询的接口和处理器

关键不是写得多,而是分得清。命令不返回业务数据(只返回 voidTask),查询不修改状态(方法体里不能有 SaveChangesUpdate 等)。所有类型都应显式声明职责:

  • ICommandHandler:只接受一个 TCommand,无返回值
  • IQueryHandler:接受 TQuery,必须返回 TResult
  • 命令和查询类型本身是 record 或不可变 class,不继承、不带行为

示例:

public record CreateOrderCommand(string CustomerId, decimal Amount);
public interface ICommandHandler
{
    Task Handle(TCommand command);
}

public class CreateOrderCommandHandler : ICommandHandler
{
    private readonly OrderDbContext _db;
    public CreateOrderCommandHandler(OrderDbContext db) => _db = db;

    public async Task Handle(CreateOrderCommand command)
    {
        var order = new Order { CustomerId = command.CustomerId, Amount = command.Amount };
        await _db.Orders.AddAsync(order);
        await _db.SaveChangesAsync();
    }
}

如何避免手写大量 if-else 或 switch 路由逻辑

不用 MediatR 就意味着没有自动泛型解析,但也不必硬写反射调度。推荐两种轻量方案:

StudyCorgi ChatGPT Detector
StudyCorgi ChatGPT Detector

StudyCorgi推出的帮助学生检测ChatGPT的工具

下载
  • DI 容器直接注册具体处理器,按需注入——比如在 Controller 里明确构造 CreateOrderCommandHandler,不追求“统一路由”
  • 若真需要统一入口(如 API 层只暴露一个 Dispatch()),可用 Dictionary 静态缓存已注册的处理器实例,首次访问时通过 Activator.CreateInstance 构建并缓存,后续直接 Cast 调用
  • 切忌在调度层做运行时类型判断 + 反射调用——性能差、堆分配多、调试困难

注意:IServiceProvider.GetService(Type) 可用,但必须确保该 Type 已在 DI 中注册为具体实现,否则返回 null 不报错,容易漏测。

查询侧容易忽略的隔离细节

很多人只拆了命令,查询仍用 EF 的 DbSet 直接暴露给 API,这等于没 CQRS。真正隔离要体现在三处:

  • 查询 DTO 必须和实体类物理分离(不同命名空间、不同程序集更佳),禁止 select new OrderDto() 之外的任何对实体的引用
  • 查询 Handler 内部应使用 AsNoTracking(),且不复用命令侧的 DbContext 实例(哪怕同一请求周期)
  • 避免在查询中调用 Include() 加载深层导航——那是命令侧或领域服务的事;查询应只投射所需字段,用 SELECT x,y,z 级别控制

一个典型错误是:在 GetOrderSummaryQueryHandler 里调用了 _db.Orders.Include(x => x.Items),结果无意中触发了延迟加载或全表扫描——这不是 CQRS,这是披着查询外衣的命令副作用。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

238

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

499

2024.03.01

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

786

2023.08.22

switch语句用法
switch语句用法

switch语句用法:1、Switch语句只能用于整数类型,枚举类型和String类型,不能用于浮点数类型和布尔类型;2、每个case语句后面必须跟着一个break语句,以防止执行其他case的代码块,没有break语句,将会继续执行下一个case的代码块;3、可以在一个case语句中匹配多个值,使用逗号分隔;4、Switch语句中的default代码块是可选的等等。

544

2023.09.21

Java switch的用法
Java switch的用法

Java中的switch语句用于根据不同的条件执行不同的代码块。想了解更多switch的相关内容,可以阅读本专题下面的文章。

424

2024.03.13

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

177

2023.11.23

java中void的含义
java中void的含义

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

102

2025.11.27

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

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

1206

2023.10.19

AO3官网入口与中文阅读设置 AO3网页版使用与访问
AO3官网入口与中文阅读设置 AO3网页版使用与访问

本专题围绕 Archive of Our Own(AO3)官网入口展开,系统整理 AO3 最新可用官网地址、网页版访问方式、正确打开链接的方法,并详细讲解 AO3 中文界面设置、阅读语言切换及基础使用流程,帮助用户稳定访问 AO3 官网,高效完成中文阅读与作品浏览。

5

2026.02.02

热门下载

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

精品课程

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

共28课时 | 3.8万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.4万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

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

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