在ASP.NET Core微服務架構下使用RabbitMQ如何實現CQRS模式

前言

在現代軟件開發中,微服務架構和CQRS模式都是備受關注的技術趨勢。微服務架構通過將應用程序拆分爲一系列小型、自治的服務,提供了更好的可伸縮性和靈活性。而CQRS模式則通過將讀操作和寫操作分離,優化了系統的性能和可維護性。本文小編將爲大家介紹如何在ASP.NET Core微服務架構下使用RabbitMQ來實現CQRS模式。

微服務架構的簡要概覽

微服務架構是一種軟件架構模式,它將一個大型的單體應用程序拆分爲一組小型、自治的服務,每個服務都可以獨立部署、擴展和管理。每個服務都專注於一個特定的業務功能,並通過輕量級的通信機制相互協作,形成一個完整的分佈式系統。

RabbitMQ在微服務中的作用

消息代理,以RabbitMQ作爲示例,是微服務架構的樞紐,爲服務間異步通信提供了一個健壯的機制。它們使得分離組件間的通信變得解耦合、可靠和可擴展。在下面的這段代碼裏面,RabbitMQ被用於給特定隊列發送消息,確保服務間通信可靠。

// Example of using RabbitMQ with RabbitMQ.Client in C#
using RabbitMQ.Client;
class RabbitMQService {
    public void SendMessageToQueue(string queueName, string message) {
        var factory = new ConnectionFactory(){HostName="localhost"};
        using var connection = factory.CreateConnection();
        using var channel = connection.CreateModel;
        channel.QueueDeclare(queue:queueName,durable:false,exclusive:false,autoDelete:false,arguments:null);
        var body=Encoding.UTF8.GetBytes(message);
        channel.BasicPublish(exchange:"",routingKey:queueName,basicProperties:null,body:body);
        Console.WriteLines($"Message sent to {queueName}:{message}");
    }
}

RabbitMQ提供了很多功能,使得針對微服務架構高度適合:

  • 可靠性:它確保消息可靠傳輸,支持消息識別機制。
  • 靈活性:支持多種消息模式(發佈訂閱,點對點)和協議(AMQP,MQTT)。
  • 可擴展:允許通過發佈橫跨不同節點或集羣的消息來橫向伸縮。

下面這段代碼演示了RabbitMQ如何實現一個發佈和訂閱的功能。

// Example of using RabbitMQ for Publish-Subscribe
public class Publisher
{
    public void Publish(string exchangeName, string message)
    {
        var factory = new ConnectionFactory() { HostName = "localhost" };
        using var connection = factory.CreateConnection();
        using var channel = connection.CreateModel();
        channel.ExchangeDeclare(exchange: exchangeName, type: ExchangeType.Fanout);
        var body = Encoding.UTF8.GetBytes(message);
        channel.BasicPublish(exchange: exchangeName, routingKey: "", basicProperties: null, body: body);
        Console.WriteLine($"Message published to {exchangeName}: {message}");
    }
}

CQRS 模式

CQRS從根本上來說是把處理命令(改變系統狀態)的職責從查詢(不更改狀態下獲取數據)中分離出來。這種分離允許對每種類型操作進行優化和裁剪。如下方的代碼所示,Command Handler(命令程序)處理寫操作,負責執行更新、創建或刪除等改變系統狀態的操作。Query Handler(查詢程序)處理讀操作,負責提供數據查詢和展示的功能。

// Example of Command and Query models in C#
public class Command {
    public string Id {get;set;}
    public object Payload{get;set}
}

public class Query {
    public string Id(get;set;)
}
// Command Handler
public class CommandHandler {
    public void HandleCommand(Command command) {
        // Logic to process and update the system state based on the command
    }
}
// Query Handler
public class QueryHandler {
    public object HandleQuery(Query query) {
        // Logic to retrieve and return data without altering the system state
        return null;
    }
}

分離讀和寫操作的優勢

  • 易於優化:不同模型可以爲它們特定的任務進行優化。
  • 可擴展:系統可以爲讀和寫獨立擴展,優化性能。
  • 靈活性:修改寫邏輯不影響讀操作,在設計和迭代上提供了更大的靈活性。
// Command Handler
public class CommandHandler {
    public void HandleCommand(Command command){
        // Logic to process and update the system state based on the command
    }
}
// Query handler
public class QueryHandler{
    public object HandlerQuery(Query query) {
        // Logic to retrieve and return data without altering the system state
        return null;
    }
}

RabbitMQ與CQRS集成

在集成CQRS與RabbitMQ時,需要考慮以下因素:

  • 消息結構:以一種清晰一致的格式爲命令和事件設計消息。
  • 錯誤處理:在消息處理中實現針對錯誤處理和重試的策略。
  • 消息持久性:配置隊列來確保消息持久,避免數據丟失。
  • 可伸縮性:通過考慮RabbitMQ集羣和負載均衡,爲可伸縮提前謀劃。

現在,小編以在線訂單系統爲場景,介紹如何集成RabbitMQ和CQRS來實現訂單的異步處理。

場景:

在一個在線訂單系統中,放置了新訂單後,它就需要被異步處理。小編將會使用RabbitMQ來處理命令(放置訂單)和事件(訂單處理)。這個系統將會用隊列來分離命令和事件,同時遵循CQRS原則。

設計注意事項:

  • OrderCommand:表示下訂單的命令。
  • OrderEvent:表示已處理的訂單。
  • Error Handling:對失敗訂單實施重試機制。

命令處理:

public class OrderCommandHandler
{
    private readonly string commandQueueName = "order_commands";

    public void SendOrderCommand(OrderCommand command)
    {
        var factory = new ConnectionFactory() { HostName = "localhost" };
        using var connection = factory.CreateConnection();
        using var channel = connection.CreateModel();
        channel.QueueDeclare(queue: commandQueueName, durable: false, exclusive: false, autoDelete: false, arguments: null);
        var body = Encoding.UTF8.GetBytes(JsonConvert.SerializeObject(command));
        channel.BasicPublish(exchange: "", routingKey: commandQueueName, basicProperties: null, body: body);
        Console.WriteLine($"Order command sent: {JsonConvert.SerializeObject(command)}");
    }
    
    public void ConsumeOrderCommands()
    {
        var factory = new ConnectionFactory() { HostName = "localhost" };
        using var connection = factory.CreateConnection();
        using var channel = connection.CreateModel();
        channel.QueueDeclare(queue: commandQueueName, durable: false, exclusive: false, autoDelete: false, arguments: null);
        var consumer = new EventingBasicConsumer(channel);
        consumer.Received += (model, ea) =>
        {
            var body = ea.Body.ToArray();
            var commandMessage = Encoding.UTF8.GetString(body);
            var orderCommand = JsonConvert.DeserializeObject<OrderCommand>(commandMessage);

            // 處理訂單命令
            Task.Run(() => ProcessOrderCommand(orderCommand));

            // 確認消息
            channel.BasicAck(deliveryTag: ea.DeliveryTag, multiple: false);
        };
        channel.BasicConsume(queue: commandQueueName, autoAck: false, consumer: consumer);
    }
    
    private void ProcessOrderCommand(OrderCommand orderCommand)
    {
        // 異步處理訂單命令的邏輯
        Console.WriteLine($"Processing order command: {JsonConvert.SerializeObject(orderCommand)}");
        
        // 下訂單,執行驗證
        // 如果成功,發佈一個訂單處理事件
        var orderEvent = new OrderEvent { OrderId = orderCommand.OrderId, Status = "Processed" };
        SendOrderProcessedEvent(orderEvent);
    }
    
    private void SendOrderProcessedEvent(OrderEvent orderEvent)
    {
        var eventQueueName = "order_events";
        var factory = new ConnectionFactory() { HostName = "localhost" };
        using var connection = factory.CreateConnection();
        using var channel = connection.CreateModel();
        channel.QueueDeclare(queue: eventQueueName, durable: false, exclusive: false, autoDelete: false, arguments: null);
        var body = Encoding.UTF8.GetBytes(JsonConvert.SerializeObject(orderEvent));
        channel.BasicPublish(exchange: "", routingKey: eventQueueName, basicProperties: null, body: body);
        Console.WriteLine($"Order processed event sent: {JsonConvert.SerializeObject(orderEvent)}");
    }
}

爲命令和事件實現消息隊列

在集成RabbitMQ的基於CQRS系統中,爲命令和事件建立的分離的隊列能使得組件間異步通信。

public class OrderEventConsumer
{
    private readonly string eventQueueName = "order_events";

    public void ConsumeOrderEvents()
    {
        var factory = new ConnectionFactory() { HostName = "localhost" };
        using var connection = factory.CreateConnection();
        using var channel = connection.CreateModel();
        channel.QueueDeclare(queue: eventQueueName, durable: false, exclusive: false, autoDelete: false, arguments: null);
        var consumer = new EventingBasicConsumer(channel);
        consumer.Received += (model, ea) =>
        {
            var body = ea.Body.ToArray();
            var eventMessage = Encoding.UTF8.GetString(body);
            var orderEvent = JsonConvert.DeserializeObject<OrderEvent>(eventMessage);
            Console.WriteLine($"Received order processed event: {JsonConvert.SerializeObject(orderEvent)}");
            // 處理已處理訂單事件的邏輯
        };
        channel.BasicConsume(queue: eventQueueName, autoAck: true, consumer: consumer);
    }
}

異步通信和事件驅動架構

事件驅動架構中,RabbitMQ使得異步通信更加便捷,這是因爲它允許組件以一種非阻塞方式對事件和消息進行響應。

public class Program
{
    public static void Main(string[] args)
    {
        var orderCommandHandler = new OrderCommandHandler();
        var orderEventConsumer = new OrderEventConsumer();

        // 舉例:發送訂單命令
        var orderCommand = new OrderCommand { OrderId = Guid.NewGuid(), Product = "Product A", Quantity = 2 };
        orderCommandHandler.SendOrderCommand(orderCommand);

        // 異步使用訂單命令和事件
        Task.Run(() => orderCommandHandler.ConsumeOrderCommands());
        Task.Run(() => orderEventConsumer.ConsumeOrderEvents());
        Console.ReadLine(); // 保持應用程序運行
    }
}

在微服務中集成CQRS和RabbitMQ

創建服務

現在小編創建兩個服務,一個用於訂單消息處理(OrderComandService),一個用於訂單查詢處理(OrderQueryService)。

OrderComandService(訂單命令服務)

// 處理命令(下訂單)
public class OrderCommandService
{
    private readonly string commandQueueName = "order_commands";
    public void SendOrderCommand(OrderCommand command)
    {
        // 向RabbitMQ隊列發送order命令的代碼(具體可以參考前面SendOrderCommand的代碼)
    }
    public void ConsumeOrderCommands()
    {
        // 從RabbitMQ隊列中消費訂單命令的代碼(具體可以參考前面的ConsumeOrderCommands代碼)
        // 異步處理接收到的命令並相應地觸發事件
    }
}

OrderQueryService(訂單查詢服務)

// 處理查詢(獲取訂單)
public class OrderQueryService
{
    private readonly string queryQueueName = "order_queries";
    public void SendOrderQuery(Query query)
    {
        // 向RabbitMQ隊列發送order命令的代碼(具體可以參考前面SendOrderCommand的代碼)
    }
    public void ConsumeOrderQueries()
    {
        // 從RabbitMQ隊列中接受消費訂單命令的代碼(具體可以參考前面的ConsumeOrderCommands代碼)
        // 異步處理接收到的查詢並檢索訂單數據
    }
}

在微服務中定義命令和查詢模型

命令和查詢模型

// 命令模型
public class OrderCommand
{
    public string OrderId { get; set; }
    // 其他與訂單相關的字段(省略)
}
// 查詢模型
public class OrderQuery
{
    public string QueryId { get; set; }
    // 其他與訂單相關的字段(省略)
}

使用RabbitMQ編寫訂單命令和訂單查詢:

OrderCommandService(訂單命令服務)

// 發送訂單命令
OrderCommandService orderCommandService = new OrderCommandService();
OrderCommand orderCommand = new OrderCommand { OrderId = "123", /* 其他訂單屬性 */ };
orderCommandService.SendOrderCommand(orderCommand);
// 消費訂單命令
orderCommandService.ConsumeOrderCommands();

OrderQueryService(訂單查詢服務)

// 發送訂單查詢
OrderQueryService orderQueryService = new OrderQueryService();
OrderQuery orderQuery = new OrderQuery { QueryId = "456", /* 其他訂單屬性 */ };
orderQueryService.SendOrderQuery(orderQuery);
// 消費訂單查詢
orderQueryService.ConsumeOrderQueries();

總結

在ASP.NET Core微服務架構中,使用RabbitMQ作爲消息隊列服務,通過實現CQRS模式(Command Query Responsibility Segregation),將寫操作和讀操作分離,以提高系統的性能和可伸縮性。這種組合能夠實現異步通信和事件驅動架構,通過將命令發送到命令處理器執行寫操作,同時使用訂閱模式將事件發佈給查詢服務,實現實時的數據查詢和更新。這樣的架構使系統更具彈性和擴展性,併爲開發者提供更好的工具和方法來構建複雜的分佈式系統,以滿足不同業務需求。

擴展鏈接:

Redis從入門到實踐

一節課帶你搞懂數據庫事務!

Chrome開發者工具使用教程

如何在Web應用中添加一個JavaScript Excel查看器

高性能渲染——詳解HTML Canvas的優勢與性能

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章