.NET Core開發實戰(第34課:MediatR:輕松實現命令查詢職責分離模式(CQRS))--學習筆記(上)


34 | MediatR:輕松實現命令查詢職責分離模式(CQRS)

核心對象

IMeditator

IRequese、IRequest

IRequestHandler<in TRequest, TResponse>

首先我們安裝了 MediatR 的 8.0 的組件包,還安裝了依賴注入框架的擴展包,以及依賴注入框架的核心組件包

  • MediatR
  • MediatR.Extensions.Microsoft.DependencyInjection
  • Microsoft.Extensions.DependencyInjection

大家可以觀察到 MediatR 的包名和命名空間少了一個 o,猜測是作者故意這樣設計的,因為它具體實現里面會有一個接口和類是 Mediator,如果設置同名的話會有一些引用上的問題

var services = new ServiceCollection();

services.AddMediatR(typeof(Program).Assembly);

我們在這里構建一個 ServiceCollection,然后通過一行代碼將我們當前的程序集注入進去,它就可以掃描我們當前程序集相關的類,下面看一下我們定義的兩個類

internal class MyCommand : IRequest<long>
{ 
    public string CommandName { get; set; }
}

internal class MyCommandHandler : IRequestHandler<MyCommand, long>
{
    public Task<long> Handle(MyCommand request, CancellationToken cancellationToken)
    {
        Console.WriteLine($"MyCommandHandler執行命令:{request.CommandName}");
        return Task.FromResult(10L);
    }
}

第一個類是 MyCommand,它實現了 IRequest 接口,這個接口就代表中介者要執行的命令

第二個類是 MyCommandHandler,它實現了 IRequestHandler 的接口,這個就是我們對命令的處理器的定義

var serviceProvider = services.BuildServiceProvider();

var mediator = serviceProvider.GetService<IMediator>();

await mediator.Send(new MyCommand { CommandName = "cmd01" });

我們從容器里面獲取一個 IMediator,然后通過 send 方法發送一個 MyCommand 命令,我們構造了一個新的 MyCommand 的實例傳給它

啟動程序,輸出如下:

MyCommandHandler執行命令:cmd01

我們可以看到 MyCommandHandler 的 Handle 方法執行了,它輸出了 MyCommandHandler 的執行命令 cmd01

這樣子,這個中介者它有什么好處呢?

大家可以看到,通過中介者模式,我們將命令的構造和命令的處理可以分離開,那么命令的處理如何知道要處理哪個命令呢,就是通過我們泛型的約束來定義的,我們這里為 IRequestHandler 填入了 MyCommand 類型,所以我們能明確知道 MyCommandHandler 是用來處理 MyCommand 的

如果說我在程序里面實現了多個 Handler,我們可以試驗一下

internal class MyCommandHandlerV2 : IRequestHandler<MyCommand, long>
{
    public Task<long> Handle(MyCommand request, CancellationToken cancellationToken)
    {
        Console.WriteLine($"MyCommandHandlerV2執行命令:{request.CommandName}");
        return Task.FromResult(10L);
    }
}

internal class MyCommandHandler : IRequestHandler<MyCommand, long>
{
    public Task<long> Handle(MyCommand request, CancellationToken cancellationToken)
    {
        Console.WriteLine($"MyCommandHandler執行命令:{request.CommandName}");
        return Task.FromResult(10L);
    }
}

啟動程序,輸出如下:

MyCommandHandlerV2執行命令:cmd01

大家可以看到我們輸出的是 V2 執行命令

我們把代碼進行一個調整,把這個定義移到后面

internal class MyCommandHandler : IRequestHandler<MyCommand, long>
{
    public Task<long> Handle(MyCommand request, CancellationToken cancellationToken)
    {
        Console.WriteLine($"MyCommandHandler執行命令:{request.CommandName}");
        return Task.FromResult(10L);
    }
}

internal class MyCommandHandlerV2 : IRequestHandler<MyCommand, long>
{
    public Task<long> Handle(MyCommand request, CancellationToken cancellationToken)
    {
        Console.WriteLine($"MyCommandHandlerV2執行命令:{request.CommandName}");
        return Task.FromResult(10L);
    }
}

啟動程序,輸出如下:

MyCommandHandler執行命令:cmd01

大家可以看到我們這次輸出的並不是 V2,而是之前的那個命令,為什么會這樣子呢?是因為實際上 mediator 對於 IRequestHandler 的掃描,它是有順序的,后面掃描到的會替換前面掃描到的 Handler,它只會識別其中最后注冊進去的一個,也就是說我們在處理 RequestHandler 的時候,我們要注意在注冊時僅注冊需要的那個

我們再來看看我們的應用程序,回到我們之前的工程里

namespace GeekTime.API.Application.Commands
{
    public class CreateOrderCommandHandler : IRequestHandler<CreateOrderCommand, long>
    {
        IOrderRepository _orderRepository;
        ICapPublisher _capPublisher;
        public CreateOrderCommandHandler(IOrderRepository orderRepository, ICapPublisher capPublisher)
        {
            _orderRepository = orderRepository;
            _capPublisher = capPublisher;
        }


        public async Task<long> Handle(CreateOrderCommand request, CancellationToken cancellationToken)
        {

            var address = new Address("wen san lu", "hangzhou", "310000");
            var order = new Order("xiaohong1999", "xiaohong", 25, address);

            _orderRepository.Add(order);
            await _orderRepository.UnitOfWork.SaveEntitiesAsync(cancellationToken);
            return order.Id;
        }
    }
}

我們可以看到我們的 CreateOrderCommandHandler 實現的是 IRequestHandler,這也就是解釋了為什么之前我們並沒有顯示的調用 CreateOrderCommandHandler,代碼卻能夠執行到這里的原因

GitHub源碼鏈接:

https://github.com/MINGSON666/Personal-Learning-Library/tree/main/DotNetCoreDevelopmentActualCombat/MediatorDemo

https://github.com/witskeeper/geektime

知識共享許可協議

本作品采用知識共享署名-非商業性使用-相同方式共享 4.0 國際許可協議進行許可。

歡迎轉載、使用、重新發布,但務必保留文章署名 鄭子銘 (包含鏈接: http://www.cnblogs.com/MingsonZheng/ ),不得用於商業目的,基於本文修改后的作品務必以相同的許可發布。

如有任何疑問,請與我聯系 (MingsonZheng@outlook.com) 。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM