Message queue là một công nghệ tốt để kết nối các thành phần trong hệ thống thông qua một máy chủ có tên là Message Broker để điều phối message. RabbitMQ là một trong những message queue rất phổ biến hiện nay. Trong bài này Dũng sẽ cùng các bạn tìm hiểu về RabbitMQ sử dụng cùng với Spring boot nhé.

Mô hình hoạt động

Đầu tiên chúng ta hãy tìm hiểu một chút về mô hình hoạt động của RabbitMQ.

Cũng có phần giống Kafka thì RabbitMQ cũng có các thành phần:

  1. Publisher: Là các kết nối từ phía các ứng dụng đẩy dữ liệu vào cho RabbitMQ Broker.
  2. Broker: Là một phần máy chủ nhận và quản lý dữ liệu được đẩy vào từ producer và đẩy dữ liệu về cho các consumer đang đăng ký nhận message từ các queue. Kiến trúc phần mềm bên trong broker bao gồm:
  • Các exchange gắn với các queue thông qua một routing key.
  • Các queue chứa các message.
  1. Consumer: Là các kết nối từ các ứng dụng để nhận và tiêu thụ dữ liệu.

Khởi tạo module

Chúng ta sẽ khởi tạo module có tên spring-boot-rabbitmq.

Chuẩn bị

Để cho đơn giản chúng ta sẽ sử dụng docker để cài đặt RabbitMQ và khởi động nó thông qua các câu lệnh sau:

docker pull rabbitmq
docker run --name=rabbitmq -p 5672:5672 -d rabbitmq:latest

Một RabbitMQ server sẽ được khởi tạo và expose cổng 5672.
Lưu ý rằng chúng ta đang sử dụng cho mục đích học tập nên sử dụng docker với RabbitMQ, trong môi trường dự án thực tế có thể RabbitMQ sẽ được cài theo một cách khác sử dụng vps hoặc máy chủ riêng biệt.

Cấu hình dự án

Chúng ta sẽ cần thay đổi tập tin spring-boot-rabbitmq/pom.xml với mã nguồn như sau:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>vn.techmaster</groupId>
        <artifactId>mastering-spring-boot</artifactId>
        <version>1.0.0</version>
    </parent>

    <artifactId>spring-boot-rabbitmq</artifactId>

    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-dependencies</artifactId>
                <version>${spring.boot.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-amqp</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
    </dependencies>
</project>

Ở đây chúng bổ sung thư viện spring-boot-starter-amqp để có thể sử dụng các thành phần liên quan đến rabbitmq client.

Cấu hình

Để có thể tạo ra đối tượng producer để kết nối và gửi được dữ liệu đến RabbitMQ broker chúng ta sẽ cần tạo ra lớp RabbitMQConfig với mã nguồn như sau:

package vn.techmaster.rabbitmq.config;

import org.springframework.amqp.core.Binding;
import org.springframework.amqp.core.BindingBuilder;
import org.springframework.amqp.core.Queue;
import org.springframework.amqp.core.TopicExchange;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class RabbitMQConfig {

    public static final String QUEUE_NAME = "techmasterQueue";
    public static final String EXCHANGE_NAME = "techmasterExchange";
    public static final String ROUTING_KEY = "techmasterRoutingKey";

    @Bean
    public Queue queue() {
        return new Queue(QUEUE_NAME, false);
    }

    @Bean
    public TopicExchange exchange() {
        return new TopicExchange(EXCHANGE_NAME);
    }

    @Bean
    public Binding binding(Queue queue, TopicExchange exchange) {
        return BindingBuilder.bind(queue).to(exchange).with(ROUTING_KEY);
    }
}

Như đã nói ở trong sơ đồ trên, thì để có thể gửi dữ liệu đến một queue thì chúng ta phải thông qua một Exchange và một routing key.

Tạo một producer

Tiếp theo chúng ta có thể tạo ra một lớp chuyên để gửi dữ liệu thế này:

package vn.techmaster.rabbitmq.producer;

import lombok.AllArgsConstructor;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.stereotype.Component;
import vn.techmaster.rabbitmq.config.RabbitMQConfig;

@Component
@AllArgsConstructor
public class RabbitMQProducer {

    private final RabbitTemplate rabbitTemplate;

    public void sendMessage(String message) {
        rabbitTemplate.convertAndSend(
            RabbitMQConfig.EXCHANGE_NAME,
            RabbitMQConfig.ROUTING_KEY,
            message
        );
        System.out.println("Sent: " + message);
    }
}

Nó sẽ gửi dữ liệu đến exchange với một routing, sau đó RabbitMQ broker sẽ tự phân bổ đến queue tương ứng.

Tạo một consumer

Để có thể tiêu thụ và giải phóng message khỏi queue chúng ta có thể tạo ra một lớp để tiêu thụ dữ liệu như sau:

package vn.techmaster.rabbitmq.consumer;

import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;
import vn.techmaster.rabbitmq.config.RabbitMQConfig;

@Component
public class RabbitMQConsumer {

    @RabbitListener(queues = RabbitMQConfig.QUEUE_NAME)
    public void handle(String message) {
        System.out.println("Handle message: " + message);
    }
}

Ở đây chúng ta đang lắng nghe dữ liệu gửi đến từ một queue.

Khởi chạy chương trình

Để khởi chạy chương trình chúng ta sẽ cần tạo ra lớp SpringBootRabbitMQStartUp với mã nguồn như sau:

package vn.techmaster.rabbitmq;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ApplicationContext;
import vn.techmaster.rabbitmq.producer.RabbitMQProducer;

@SpringBootApplication
public class SpringBootRabbitMQStartUp {

    public static void main(String[] args) {
        ApplicationContext applicationContext = SpringApplication
            .run(SpringBootRabbitMQStartUp.class);
        RabbitMQProducer messageProducer = applicationContext
            .getBean(RabbitMQProducer.class);
        messageProducer.sendMessage("Hello Techmaster");
    }
}

Ở đây chúng ta lấy ra đối tượng messageProducer để gửi đến RabbitMQ broker dữ liệu “Hello Techmaster”.
Consumer sẽ nhận được message từ queue và tiến hành xử lý. kết quả chúng ta nhận được là:

// bỏ qua các log trước đó
Sent: Hello Techmaster
Handle message: Hello Techmaster

Tổng kết

Như vậy chúng ta đã cùng nhau tìm hiểu spring boot rabbitmq, tìm hiểu cách để tạo ra producer, consumer và chạy thử thành công chương trình.


Cám ơn bạn đã quan tâm đến bài viết|video này. Để nhận được thêm các kiến thức bổ ích bạn có thể:

  1. Đọc các bài viết của TechMaster trên facebook: https://www.facebook.com/techmastervn
  2. Xem các video của TechMaster qua Youtube: https://www.youtube.com/@TechMasterVietnam nếu bạn thấy video/bài viết hay bạn có thể theo dõi kênh của TechMaster để nhận được thông báo về các video mới nhất nhé.
  3. Chat với techmaster qua Discord: https://discord.gg/yQjRTFXb7a