C# 如何使用MediatR库 – 实现CQRS模式中的命令和查询

CQRS将操作分为命令与查询,MediatR通过中介者模式实现解耦,提升系统可维护性与扩展性,适用于复杂业务场景。

c# 如何使用mediatr库 - 实现cqrs模式中的命令和查询

在现代C#应用程序开发中,CQRS(命令查询职责分离)模式被广泛用于提升系统可维护性和可扩展性。MediatR 是一个轻量级库,帮助我们在项目中轻松实现这一模式。它通过中介者模式将请求与处理逻辑解耦,使代码更清晰、更易于测试。

什么是CQRS?

CQRS 将数据操作分为两类:

命令(Commands):用于修改状态的操作,比如创建、更新或删除数据。 查询(Queries):仅用于读取数据,不改变任何状态。 这种分离让我们可以为写操作和读操作设计不同的模型、数据库甚至架构。

安装与配置 MediatR

在使用之前,先通过 NuGet 安装 MediatR 和依赖注入支持包:

dotnet add package MediatR
dotnet add package MediatR.Extensions.Microsoft.DependencyInjection

然后在 Program.csStartup.cs 中注册服务:

builder.Services.AddMediatR(cfg => cfg.RegisterServicesFromAssembly(typeof(Program).Assembly));

这样,MediatR 就会自动发现并注册所有实现 IRequestHandler 的类。

定义命令与处理程序

以“创建用户”为例,我们定义一个命令类:

public record CreateUserCommand(string Name, string Email) : IRequest;

接着编写对应的处理程序:

public class CreateUserCommandHandler : IRequestHandler
{
private readonly IUserRepository _userRepository;

public CreateUserCommandHandler(IUserRepository userRepository)
{
_userRepository = userRepository;
}

public async Task Handle(CreateUserCommand request, CancellationToken ct)
{
var user = new User(request.Name, request.Email);
await _userRepository.AddAsync(user, ct);
return user.Id;
}
}

这个处理程序接收命令,执行业务逻辑,并返回新用户的 ID。

定义查询与处理程序

对于查询,比如“根据ID获取用户信息”,我们这样定义:

public record GetUserByIdQuery(Guid Id) : IRequest;

处理程序从只读数据源中提取数据:

public class GetUserByIdQueryHandler : IRequestHandler
{
private readonly IUserReadRepository _readRepository;

public GetUserByIdQueryHandler(IUserReadRepository readRepository)
{
_readRepository = readRepository;
}

public async Task Handle(GetUserByIdQuery request, CancellationToken ct)
{
var user = await _readRepository.GetByIdAsync(request.Id, ct);
if (user == null) throw new KeyNotFoundException();
return user;
}
}

在控制器中使用 MediatR

在 ASP.NET Core 控制器中注入 ISender 接口来发送请求:

[ApiController]
[Route(“api/users”)]
public class UsersController : ControllerBase
{
private readonly ISender _sender;

public UsersController(ISender sender)
{
_sender = sender;
}

[HttpPost]
public async Task CreateUser([FromBody] CreateUserCommand command)
{
var userId = await _sender.Send(command);
return CreatedAtAction(nameof(GetUser), new { id = userId }, userId);
}

[HttpGet(“{id}”)]
public async Task> GetUser(Guid id)
{
var query = new GetUserByIdQuery(id);
var user = await _sender.Send(query);
return Ok(user);
}
}

ISender 是 MediatR 提供的核心接口,可用于发送任意请求类型。

优点与适用场景

使用 MediatR 实现 CQRS 带来的好处包括:

职责清晰:每个类只做一件事。 便于添加横切关注点:如日志、验证、事务控制等,可通过行为(Behaviors)统一处理。 可测试性强:处理程序容易单元测试。 适合复杂业务系统:尤其在读写频率差异大或性能要求高的场景下表现优异。

基本上就这些。掌握 MediatR 的基本用法后,你可以逐步引入管道行为、缓存机制或事件发布等功能,进一步增强系统的灵活性和健壮性。不复杂但容易忽略的是保持请求类的简洁和单一职责,避免滥用导致过度拆分。

以上就是C# 如何使用MediatR库 – 实现CQRS模式中的命令和查询的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1443042.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 19:40:00
下一篇 2025年12月17日 04:06:56

相关推荐

  • C# XML解析终极排查步骤 快速定位问题的5个核心技巧

    掌握C# XML解析需遵循五个核心技巧:1. 验证XML格式合法性,使用XDocument.Load或XElement.Parse捕获XmlException定位错误;2. 检查命名空间定义并用XNamespace声明,避免查询失效;3. 采用安全访问如null条件判断或空合并操作符防止NullRe…

    好文分享 2025年12月17日
    000
  • Blazor IDisposable 接口实现资源清理

    Blazor组件必须实现IDisposable以释放非托管资源,如Timer、事件订阅、WebSocket等;应在Dispose()中判空调用Dispose(),避免StateHasChanged()和await异步操作。 Blazor 组件实现 IDisposable 是为了在组件被销毁前及时释放…

    2025年12月17日
    000
  • Blazor Streaming Rendering 使用教程

    Blazor Streaming Rendering是.NET 8引入的流式渲染优化,通过@stream指令分块推送渲染结果,提升长列表和实时数据场景的响应体验。 Blazor Streaming Rendering(流式渲染)是 .NET 8 引入的一项关键优化,用于提升 Blazor Serve…

    2025年12月17日
    000
  • Blazor WebAssembly 和 Blazor Server 混合应用怎么做

    Blazor混合应用本质是共享代码+分离宿主,需通过Shared类库共用组件与逻辑,WASM与Server项目独立部署并适配差异,不可单项目切换渲染模式。 Blazor WebAssembly 和 Blazor Server 混合应用不是官方直接支持的“单项目双托管模式”,但可以通过合理架构设计,在…

    2025年12月17日
    000
  • Blazor 组件化 CSS 作用域隔离教程

    Blazor 的 CSS 隔离是编译期自动为 .razor.css 文件中选择器添加唯一属性标识并注入对应 HTML 属性,实现组件级样式作用域;需同名同目录配对文件,支持 ::deep 透传和 :global() 全局声明。 Blazor 提供了原生的 CSS 隔离(CSS Isolation)机…

    2025年12月17日
    100
  • Blazor JWT Token 登录和授权方法

    Blazor中JWT登录授权需前后端闭环协作:后端用HMAC-SHA256签发含name/role/exp的token并返回UserToken结构;前端存localStorage、HttpClient自动携Bearer头、AuthorizeView拦截路由、401时跳登录页、登出时清除token并重…

    2025年12月17日
    000
  • Blazor 怎么使用 RenderTreeBuilder 手动构建UI

    RenderTreeBuilder 是 Blazor 中用于手动构建渲染树的底层 API,需在 BuildRenderTree 中同步调用,通过 OpenElement/OpenComponent/AddAttribute 等方法生成轻量中间表示,严格配对且序号递增,适用于虚拟滚动等高性能场景,但非…

    2025年12月17日
    000
  • C#怎么发起HTTP请求 C# HttpClient使用方法详解

    推荐使用HttpClient发起HTTP请求,它线程安全、支持异步且可复用;应全局复用实例避免端口耗尽;GET用GetStringAsync,POST用StringContent发送JSON;需设置请求头、超时并手动处理响应状态与异常。 在C#中发起HTTP请求,推荐且现代的方式是使用 HttpCl…

    2025年12月17日
    000
  • ASP.NET Core怎么实现JWT认证 JWT Token生成与验证方法

    ASP.NET Core实现JWT认证需配置Bearer方案、生成Token并启用中间件,关键在密钥一致、时间对齐、Claim明确、传输合规;需注册认证服务、手动签发Token、正确启用UseAuthentication/UseAuthorization,并注意安全细节如避免敏感信息入载荷、强制HT…

    2025年12月17日
    000
  • C#怎么使用Polly实现重试和熔断 Polly弹性框架使用教程

    Polly 是 .NET 中主流弹性容错库,支持直观配置重试、熔断及组合策略。通过 Policy.WrapAsync 组合重试与熔断,配合指数退避、jitter、日志回调和 PolicyRegistry 等特性实现高可用。 Polly 是 .NET 生态中最主流的弹性与容错库,用它实现重试(Retr…

    2025年12月17日
    200
  • Dapper.FluentMap是什么 Dapper.FluentMap流畅映射配置教程

    Dapper.FluentMap 是一个为 Dapper 设计的轻量级编译期映射配置库,通过 Fluent API 在启动时注册列名映射规则,解决数据库下划线命名、字段前缀/后缀、审计字段忽略及统一约定等场景,需在首次查询前完成 Initialize 初始化。 Dapper.FluentMap 是什…

    2025年12月17日
    000
  • Blazor 依赖注入(DI)配置和使用方法

    Blazor依赖注入是框架级基础设施,服务需在Program.cs中按生命周期注册(Singleton/Scoped/Transient),组件用@inject或构造函数注入,自定义服务须接口+实现+注册闭环,Scoped在WebAssembly中按标签页隔离。 Blazor 依赖注入(DI)不是可…

    2025年12月17日
    000
  • Ant Design of Blazor 快速上手指南

    Ant Design of Blazor 是基于 Ant Design 规范、适配 Blazor Server/WASM 的 UI 组件库;需通过 NuGet 安装、Program.cs 注册服务、引入 CSS 样式,并在 Razor 页面中使用 @using AntDesign 和 等组件快速开发…

    2025年12月17日
    000
  • C#如何进行数据库分片 ShardingSphere-Proxy .NET客户端用法

    ShardingSphere-Proxy 作为兼容 MySQL/PostgreSQL 协议的透明代理,.NET 应用只需使用 MySqlConnector 或 Npgsql 等标准 ADO.NET 驱动连接其地址(如 127.0.0.1:3307),即可透明执行分片路由,无需官方 SDK 或修改业务…

    2025年12月17日
    000
  • C#如何使用nameof表达式 C# nameof运算符使用技巧

    nameof 是 C# 6.0 引入的编译时运算符,将标识符转换为字符串字面量,支持重构、避免拼写错误,适用于变量、属性、方法、类型等,但不支持表达式或动态名称。 nameof 是 C# 6.0 引入的编译时运算符,它不执行任何运行时操作,只在编译期将标识符(如变量名、方法名、属性名、类型名等)转换…

    2025年12月17日
    000
  • C#如何实现分布式锁 Redis分布式锁C#实现方法

    C#中实现Redis分布式锁需用SET命令原子加锁并设唯一value,通过Lua脚本安全解锁防误删,配合Watchdog机制自动续期,再辅以重连、重试、日志等生产级保障。 在C#中实现Redis分布式锁,核心是利用Redis的SET命令原子性设置带过期时间的key,并配合唯一value(如GUID)…

    2025年12月17日
    000
  • C#怎么实现单例模式 C#设计模式之单例实现方法

    推荐使用Lazy实现单例模式,线程安全且支持延迟初始化;其次可选静态构造函数方式(非延迟);DCL仅用于旧框架兼容。需注意避免滥用、优先考虑DI容器替代。 在C#中实现单例模式,核心是确保一个类只有一个实例,并提供全局访问点。最常用、最推荐的是静态构造函数 + 私有静态只读字段方式(线程安全、简洁、…

    2025年12月17日
    000
  • Blazor 弹窗(Modal)怎么实现

    Blazor纯C#实现Modal弹窗无需JS,核心是状态驱动显隐、防滚动、点击遮罩/ESC关闭、焦点管理及可配置样式;通过RenderFragment支持嵌套内容,配合CSS遮罩与动画,兼顾可访问性与体验。 Blazor 实现弹窗(Modal)不依赖 JS,纯 C# + Razor 就能搞定,核心是…

    2025年12月17日 好文分享
    000
  • C#怎么获取当前路径 C#获取程序运行目录方法

    最常用且安全的方式是使用AppDom%ignore_a_1%n.CurrentDomain.BaseDirectory获取exe所在目录;Environment.CurrentDirectory返回当前工作目录但可能变化;跨平台推荐AppContext.BaseDirectory或Assembly.…

    2025年12月17日
    100
  • C#怎么创建抽象类 C# abstract class与virtual方法

    抽象类用abstract声明,不可实例化,只能继承;可含已实现成员和必须由子类重写的abstract成员;含abstract成员的类必须声明为abstract;子类须override所有abstract成员,除非自身也声明为abstract。 在C#中,抽象类用 abstract 关键字声明,不能被…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信