Bootstrap

RabbitMQ的Topic模型学习

Topic 模型是 RabbitMQ 的高级模型之一,Topic 模型使用了通配符的概念,可以匹配更灵活的路由规则。topic模式相当于是对路由模式的一个升级,topic模式主要就是在匹配的规则上可以实现模糊匹配。

在 Topic 模型中,生产者将消息发送到交换机,交换机根据消息的 routing key 将消息转发到对应的队列中。与 Direct 模型不同的是,Topic 模型中 routing key 支持通配符匹配,其中 '*' 可以匹配一个单词,'#' 可以匹配多个单词。例如,"topic.*" 可以匹配 "topic.create","topic.delete" 等消息,而 "topic.#" 可以匹配 "topic.create.one","topic.delete.two" 等消息。

1.引入依赖

<!--rabbitmq-->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-amqp</artifactId>
</dependency>

2.创建生产者

/**
 * @author qx
 * @date 2023-04-07
 * @Descripion: 生产者 控制层
 */
@RestController
@RequestMapping("/producer")
public class ProducerController {

    // 交换机名称
    private static final String EXCHANGE_NAME = "exchange_topic";
    // 定义匹配单个单词的路由key
    private static final String EXCHANGE_ROUTING_KEY1 = "topic.km";
    // 定义匹配多个单词的路由key
    private static final String EXCHANGE_ROUTING_KEY2 = "topic.km.001";


    @RequestMapping
    public void createMessage() throws IOException, TimeoutException {
        // 获取连接
        Connection connection = ConnectionUtil.getConnection();
        if (connection == null) {
            return;
        }
        // 创建数据传输通道
        Channel channel = connection.createChannel();
        // 声明交换机 设置路由模型(direct)
        channel.exchangeDeclare(EXCHANGE_NAME, "topic");
        // 发送消息到交换机
        for (int i = 0; i < 100; i++) {
            if (i % 2 == 0) {
                // 发送消息到只匹配一个单词的路由
                channel.basicPublish(EXCHANGE_NAME, EXCHANGE_ROUTING_KEY1, MessageProperties.PERSISTENT_TEXT_PLAIN, ("topic模式发送的第 " + i + "条消息").getBytes());
            } else {
                // 发送消息到匹配多个单词的路由
                channel.basicPublish(EXCHANGE_NAME, EXCHANGE_ROUTING_KEY2, MessageProperties.PERSISTENT_TEXT_PLAIN, ("topic模式发送的第 " + i + "条消息").getBytes());
            }
        }
        //  关闭通道
        channel.close();
        // 关闭连接
        connection.close();
    }
}

3.创建两个消费者

/**
 * @author qinxun
 * @date 2023-04-07
 * @Descripion: 消费者1 控制层
 */
@RestController
@RequestMapping("/consumer1")
public class Consumer1Controller {

    // 交换机名称
    private static final String EXCHANGE_NAME = "exchange_topic";
    // 队列名称
    private static final String QUEUE_NAME = "queue_topic_1";
    // 定义路由的key 可匹配topic后面的一个单词 topic.km
    private static final String EXCHANGE_ROUTING_KEY = "topic.*";

    @RequestMapping
    public void consumerMessage1() throws IOException {
        // 获取连接
        Connection connection = ConnectionUtil.getConnection();
        if (connection == null) {
            return;
        }
        // 创建数据传输通道
        Channel channel = connection.createChannel();
        // 声明队列
        // 参数1:队列的名字
        // 参数2:是否持久化 比如现在发送到队列里面的消息,如果没有持久化,重启这个队列后数 据会丢失(false) true:重启之后数据依然在
        // 参数3:是否排外 连接关闭之后 这个队列是否自动删除(false:不自动删除) 是否允许其他通道来进行访问这个数据(false:不允许)
        // 参数4:是否自动删除 就是当最后一个连接断开的时候,是否自动删除这个队列(false:不删除)
        // 参数5:附带参数 声明队列的时候,附带的一些参数
        channel.queueDeclare(QUEUE_NAME, false, false, false, null);
        // 声明交换机
        // 参数1:交换机名称
        // 参数2:交换机类型 direct(路由模式)、fanout(发布订阅模式)、topic(topic模式-模糊匹配)、headers(标头交换,由Headers的参数分配,不常用)
        // 参数3:durable,是否持久化交换机   false:默认值,不持久化
        // 参数4:autoDelete,没有消费者使用时,是否自动删除交换机   false:默认值,不删除
        // 参数5:internal,是否内置,如果设置 为true,则表示是内置的交换器, 客户端程序无法直接发送消息到这个交换器中, 只能通过交换器路由到交换器的方式  false:默认值,允许外部直接访问
        // 参数6:arguments,交换机的一些其他属性,默认值为 null
        channel.exchangeDeclare(EXCHANGE_NAME, "topic");
        // 将队列绑定到交换机
        // 参数1:队列的名字
        // 参数2:交换机的名字
        // 参数3:路由
        channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, EXCHANGE_ROUTING_KEY);
        // 声明消费者
        DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                System.out.println("消费者1接收到的消息是:" + new String(body));
            }
        };
        // 绑定消费者
        channel.basicConsume(QUEUE_NAME, true, defaultConsumer);
    }
}
/**
 * @author qx
 * @date 2023-04-07
 * @Descripion: 消费者2 控制层
 */
@RestController
@RequestMapping("/consumer2")
public class Consumer2Controller {

    // 交换机名称
    private static final String EXCHANGE_NAME = "exchange_topic";
    // 队列名称
    private static final String QUEUE_NAME = "queue_topic_2";
    // 定义路由的key 可匹配topic后面的多个单词 topic.km.001 和topic.km
    private static final String EXCHANGE_ROUTING_KEY = "topic.#";

    @RequestMapping
    public void consumerMessage1() throws IOException {
        // 获取连接
        Connection connection = ConnectionUtil.getConnection();
        if (connection == null) {
            return;
        }
        // 创建数据传输通道
        Channel channel = connection.createChannel();
        // 声明队列
        // 参数1:队列的名字
        // 参数2:是否持久化 比如现在发送到队列里面的消息,如果没有持久化,重启这个队列后数 据会丢失(false) true:重启之后数据依然在
        // 参数3:是否排外 连接关闭之后 这个队列是否自动删除(false:不自动删除) 是否允许其他通道来进行访问这个数据(false:不允许)
        // 参数4:是否自动删除 就是当最后一个连接断开的时候,是否自动删除这个队列(false:不删除)
        // 参数5:附带参数 声明队列的时候,附带的一些参数
        channel.queueDeclare(QUEUE_NAME, false, false, false, null);
        // 声明交换机
        // 参数1:交换机名称
        // 参数2:交换机类型 direct(路由模式)、fanout(发布订阅模式)、topic(topic模式-模糊匹配)、headers(标头交换,由Headers的参数分配,不常用)
        // 参数3:durable,是否持久化交换机   false:默认值,不持久化
        // 参数4:autoDelete,没有消费者使用时,是否自动删除交换机   false:默认值,不删除
        // 参数5:internal,是否内置,如果设置 为true,则表示是内置的交换器, 客户端程序无法直接发送消息到这个交换器中, 只能通过交换器路由到交换器的方式  false:默认值,允许外部直接访问
        // 参数6:arguments,交换机的一些其他属性,默认值为 null
        channel.exchangeDeclare(EXCHANGE_NAME, "topic");
        // 将队列绑定到交换机
        // 参数1:队列的名字
        // 参数2:交换机的名字
        // 参数3:路由
        channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, EXCHANGE_ROUTING_KEY);
        // 声明消费者
        DefaultConsumer defaultConsumer = new DefaultConsumer(channel) {
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                System.out.println("消费者2接收到的消息是:" + new String(body));
            }
        };
        // 绑定消费者
        channel.basicConsume(QUEUE_NAME, true, defaultConsumer);
    }
}

4.测试

先启动2个消费者,再启动生产者

消费者1订阅的是 "order.*" 的消息,消费者2订阅的是 "order.#" 的消息,可以得到以下结果:

消费者1接收到的消息是:"Topic 模型发送的偶数条消息"

消费者2接收到的消息是:"Topic 模型发送的全部消息"

消费者2接收到的消息是:topic模式发送的第 0条消息
消费者1接收到的消息是:topic模式发送的第 0条消息
消费者2接收到的消息是:topic模式发送的第 1条消息
消费者1接收到的消息是:topic模式发送的第 2条消息
消费者2接收到的消息是:topic模式发送的第 2条消息
消费者1接收到的消息是:topic模式发送的第 4条消息
消费者2接收到的消息是:topic模式发送的第 3条消息
消费者2接收到的消息是:topic模式发送的第 4条消息
消费者1接收到的消息是:topic模式发送的第 6条消息
消费者2接收到的消息是:topic模式发送的第 5条消息
消费者2接收到的消息是:topic模式发送的第 6条消息
消费者2接收到的消息是:topic模式发送的第 7条消息
消费者2接收到的消息是:topic模式发送的第 8条消息
消费者1接收到的消息是:topic模式发送的第 8条消息
消费者2接收到的消息是:topic模式发送的第 9条消息
消费者1接收到的消息是:topic模式发送的第 10条消息
消费者1接收到的消息是:topic模式发送的第 12条消息
消费者1接收到的消息是:topic模式发送的第 14条消息
消费者1接收到的消息是:topic模式发送的第 16条消息
消费者1接收到的消息是:topic模式发送的第 18条消息
消费者1接收到的消息是:topic模式发送的第 20条消息
消费者1接收到的消息是:topic模式发送的第 22条消息
消费者1接收到的消息是:topic模式发送的第 24条消息
消费者1接收到的消息是:topic模式发送的第 26条消息
消费者1接收到的消息是:topic模式发送的第 28条消息
消费者2接收到的消息是:topic模式发送的第 10条消息
消费者1接收到的消息是:topic模式发送的第 30条消息
消费者2接收到的消息是:topic模式发送的第 11条消息
消费者2接收到的消息是:topic模式发送的第 12条消息
消费者2接收到的消息是:topic模式发送的第 13条消息
消费者2接收到的消息是:topic模式发送的第 14条消息
消费者2接收到的消息是:topic模式发送的第 15条消息
消费者2接收到的消息是:topic模式发送的第 16条消息
消费者2接收到的消息是:topic模式发送的第 17条消息
消费者1接收到的消息是:topic模式发送的第 32条消息
消费者1接收到的消息是:topic模式发送的第 34条消息
消费者1接收到的消息是:topic模式发送的第 36条消息
消费者1接收到的消息是:topic模式发送的第 38条消息
消费者2接收到的消息是:topic模式发送的第 18条消息
消费者2接收到的消息是:topic模式发送的第 19条消息
消费者2接收到的消息是:topic模式发送的第 20条消息
消费者1接收到的消息是:topic模式发送的第 40条消息
消费者1接收到的消息是:topic模式发送的第 42条消息
消费者1接收到的消息是:topic模式发送的第 44条消息
消费者2接收到的消息是:topic模式发送的第 21条消息
消费者1接收到的消息是:topic模式发送的第 46条消息
消费者2接收到的消息是:topic模式发送的第 22条消息
消费者2接收到的消息是:topic模式发送的第 23条消息
消费者2接收到的消息是:topic模式发送的第 24条消息
消费者2接收到的消息是:topic模式发送的第 25条消息
消费者2接收到的消息是:topic模式发送的第 26条消息
消费者2接收到的消息是:topic模式发送的第 27条消息
消费者2接收到的消息是:topic模式发送的第 28条消息
消费者1接收到的消息是:topic模式发送的第 48条消息
消费者2接收到的消息是:topic模式发送的第 29条消息
消费者2接收到的消息是:topic模式发送的第 30条消息
消费者2接收到的消息是:topic模式发送的第 31条消息
消费者2接收到的消息是:topic模式发送的第 32条消息
消费者2接收到的消息是:topic模式发送的第 33条消息
消费者2接收到的消息是:topic模式发送的第 34条消息
消费者2接收到的消息是:topic模式发送的第 35条消息
消费者2接收到的消息是:topic模式发送的第 36条消息
消费者1接收到的消息是:topic模式发送的第 50条消息
消费者1接收到的消息是:topic模式发送的第 52条消息
消费者1接收到的消息是:topic模式发送的第 54条消息
消费者2接收到的消息是:topic模式发送的第 37条消息
消费者2接收到的消息是:topic模式发送的第 38条消息
消费者2接收到的消息是:topic模式发送的第 39条消息
消费者2接收到的消息是:topic模式发送的第 40条消息
消费者2接收到的消息是:topic模式发送的第 41条消息
消费者2接收到的消息是:topic模式发送的第 42条消息
消费者2接收到的消息是:topic模式发送的第 43条消息
消费者2接收到的消息是:topic模式发送的第 44条消息
消费者2接收到的消息是:topic模式发送的第 45条消息
消费者2接收到的消息是:topic模式发送的第 46条消息
消费者2接收到的消息是:topic模式发送的第 47条消息
消费者2接收到的消息是:topic模式发送的第 48条消息
消费者2接收到的消息是:topic模式发送的第 49条消息
消费者2接收到的消息是:topic模式发送的第 50条消息
消费者2接收到的消息是:topic模式发送的第 51条消息
消费者2接收到的消息是:topic模式发送的第 52条消息
消费者1接收到的消息是:topic模式发送的第 56条消息
消费者1接收到的消息是:topic模式发送的第 58条消息
消费者1接收到的消息是:topic模式发送的第 60条消息
消费者1接收到的消息是:topic模式发送的第 62条消息
消费者1接收到的消息是:topic模式发送的第 64条消息
消费者1接收到的消息是:topic模式发送的第 66条消息
消费者1接收到的消息是:topic模式发送的第 68条消息
消费者1接收到的消息是:topic模式发送的第 70条消息
消费者1接收到的消息是:topic模式发送的第 72条消息
消费者2接收到的消息是:topic模式发送的第 53条消息
消费者2接收到的消息是:topic模式发送的第 54条消息
消费者2接收到的消息是:topic模式发送的第 55条消息
消费者2接收到的消息是:topic模式发送的第 56条消息
消费者2接收到的消息是:topic模式发送的第 57条消息
消费者2接收到的消息是:topic模式发送的第 58条消息
消费者2接收到的消息是:topic模式发送的第 59条消息
消费者2接收到的消息是:topic模式发送的第 60条消息
消费者2接收到的消息是:topic模式发送的第 61条消息
消费者2接收到的消息是:topic模式发送的第 62条消息
消费者2接收到的消息是:topic模式发送的第 63条消息
消费者2接收到的消息是:topic模式发送的第 64条消息
消费者2接收到的消息是:topic模式发送的第 65条消息
消费者2接收到的消息是:topic模式发送的第 66条消息
消费者2接收到的消息是:topic模式发送的第 67条消息
消费者2接收到的消息是:topic模式发送的第 68条消息
消费者1接收到的消息是:topic模式发送的第 74条消息
消费者1接收到的消息是:topic模式发送的第 76条消息
消费者1接收到的消息是:topic模式发送的第 78条消息
消费者2接收到的消息是:topic模式发送的第 69条消息
消费者2接收到的消息是:topic模式发送的第 70条消息
消费者2接收到的消息是:topic模式发送的第 71条消息
消费者2接收到的消息是:topic模式发送的第 72条消息
消费者2接收到的消息是:topic模式发送的第 73条消息
消费者2接收到的消息是:topic模式发送的第 74条消息
消费者1接收到的消息是:topic模式发送的第 80条消息
消费者1接收到的消息是:topic模式发送的第 82条消息
消费者1接收到的消息是:topic模式发送的第 84条消息
消费者2接收到的消息是:topic模式发送的第 75条消息
消费者2接收到的消息是:topic模式发送的第 76条消息
消费者2接收到的消息是:topic模式发送的第 77条消息
消费者2接收到的消息是:topic模式发送的第 78条消息
消费者2接收到的消息是:topic模式发送的第 79条消息
消费者2接收到的消息是:topic模式发送的第 80条消息
消费者2接收到的消息是:topic模式发送的第 81条消息
消费者2接收到的消息是:topic模式发送的第 82条消息
消费者2接收到的消息是:topic模式发送的第 83条消息
消费者2接收到的消息是:topic模式发送的第 84条消息
消费者2接收到的消息是:topic模式发送的第 85条消息
消费者2接收到的消息是:topic模式发送的第 86条消息
消费者2接收到的消息是:topic模式发送的第 87条消息
消费者2接收到的消息是:topic模式发送的第 88条消息
消费者2接收到的消息是:topic模式发送的第 89条消息
消费者2接收到的消息是:topic模式发送的第 90条消息
消费者2接收到的消息是:topic模式发送的第 91条消息
消费者2接收到的消息是:topic模式发送的第 92条消息
消费者2接收到的消息是:topic模式发送的第 93条消息
消费者2接收到的消息是:topic模式发送的第 94条消息
消费者2接收到的消息是:topic模式发送的第 95条消息
消费者2接收到的消息是:topic模式发送的第 96条消息
消费者2接收到的消息是:topic模式发送的第 97条消息
消费者2接收到的消息是:topic模式发送的第 98条消息
消费者2接收到的消息是:topic模式发送的第 99条消息
消费者1接收到的消息是:topic模式发送的第 86条消息
消费者1接收到的消息是:topic模式发送的第 88条消息
消费者1接收到的消息是:topic模式发送的第 90条消息
消费者1接收到的消息是:topic模式发送的第 92条消息
消费者1接收到的消息是:topic模式发送的第 94条消息
消费者1接收到的消息是:topic模式发送的第 96条消息
消费者1接收到的消息是:topic模式发送的第 98条消息

;