• Rabbitmq的使用四_Java Client方式使用Topic模式


    Rabbitmq的使用四_Java Client方式使用Topic模式

    1.官方文档地址:https://www.rabbitmq.com/tutorials/tutorial-five-java.html

    2.定义:topic和direct类似,也是将消息发送到RoutingKey和BindingKey相匹配的队列中,只不过可以模糊匹配。

    RoutinKey为一个被“.”号分割的字符串

    交换机和队列的绑定键也是一样的格式。

    topic交换机发送给队列的消息,通过routingkey进行匹配。匹配上的队列都可以收到消息。

    有两种重要的绑定键的特殊情况:

      * 号只能代替一个单词。
      # 号可以替换零个或多个单词。

                            图一                                                图二

    如图二所示:

    在这个例子中,我们将发送所有描述动物的信息。消息将与一个routingkey一起发送,routingkey由三个单词(两个点)组成。路由key中的第一个词将描述速度,第二个词描述颜色,第三个词描述物种:

    <speed>.<colour>.<species>

    Q1队列和交换机之间通过 *.orange.* 绑定键进行绑定。Topic交换机X和Q2的绑定键有两个,分别是 *.*.rabbit   和lazy.#

    这些绑定可以总结为:

      Q1队列对所有橙色的动物都感兴趣。

      Q2队列想收到关于兔子的一切消息,以及关于懒惰动物的一切消息。

    主题交换机功能强大,可以像其他交换机一样工作。

    模拟以下routintkey来验证topic交换机的使用

    quick.orange.rabbit,将被传递到两个队列。
    lazy.orange.elephant 将被传递到两个队列
    quick.orange.fox 只会去第一个队列
    lazy.brown.fox 只去第二个队列
    lazy.pink.rabbit 只会传递到第二个队列一次,即使它匹配两个绑定
    quick.brown.fox 不匹配任何绑定,因此将被丢弃
    orange 或者 quick.orange.male.rabbit   这些消息不会匹配任何绑定,将会丢失
    lazy.orange.male.rabbit 即使它有四个单词,也将匹配最后一个绑定,并将被传递到第二个队列。 

    代码案例:

    1.创建一个生产者

    public class RabbitMQtopicSender {
    
        // 创建一个topic交换机
        private static final String topic_EXCHANGE_NAME = "topic_exchange_name";
    
        public static void main(String[] args) throws IOException, TimeoutException {
            // 1.获取一个rabbitmq的连接
            Connection rabbitMQConnections = RabbitMQConnectionFactory.getRabbitMQConnections();
            // 2.创建一个通道
            Channel channel = rabbitMQConnections.createChannel();
            // 3. 声明一个topic交换机
            channel.exchangeDeclare(topic_EXCHANGE_NAME, "topic");
            // 4.向交换机发送一条消息routingkey 为:quick.orange.rabbit
            channel.basicPublish(topic_EXCHANGE_NAME, "quick.orange.rabbit", null, "quick.orange.rabbit========1=======>根据绑定键,将被传递到两个队列".getBytes());
            // 向交换机发送一条消息routingkey 为:quick.orange.rabbit
            channel.basicPublish(topic_EXCHANGE_NAME, "lazy.orange.elephant", null, "lazy.orange.elephant========2=======>根据绑定键,将被传递到两个队列".getBytes());
            channel.basicPublish(topic_EXCHANGE_NAME, "quick.orange.fox", null, "quick.orange.fox=====3=======>根据绑定键,将被传递到Q1队列".getBytes());
            channel.basicPublish(topic_EXCHANGE_NAME, "lazy.brown.fox", null, "lazy.brown.fox=======4=======>根据绑定键,将被传递到Q2队列".getBytes());
            channel.basicPublish(topic_EXCHANGE_NAME, "lazy.pink.rabbit", null, "lazy.pink.rabbit=====5======>只会传递到第二个队列一次,即使它匹配两个绑定".getBytes());
            channel.basicPublish(topic_EXCHANGE_NAME, "quick.brown.fox", null, "quick.brown.fox===6=========>匹配不到任何队列,被丢弃".getBytes());
            channel.basicPublish(topic_EXCHANGE_NAME, "orange", null, "orange=======7========>匹配不到任何队列,被丢弃".getBytes());
            channel.basicPublish(topic_EXCHANGE_NAME, "quick.orange.male.rabbit", null, "quick.orange.male.rabbit=======8======>匹配不到任何队列,被丢弃".getBytes());
            channel.basicPublish(topic_EXCHANGE_NAME, "lazy.orange.male.rabbit", null, "lazy.orange.male.rabbit=======9=======>匹配队列2".getBytes());
            System.out.println("消息发送成功");
            // 5.发送完消息后关闭连接
            channel.close();
            rabbitMQConnections.close();
        }
    }

    创建消费者1

    public class RabbitMQtopicReceiver {
        // 创建一个topic交换机
        private static final String topic_EXCHANGE_NAME = "topic_exchange_name";
        // 创建队列一
        private static final String topic_QUEUE_NUM_ONE = "topic_queue_num_one";
    
        public static void main(String[] args) throws IOException, TimeoutException {
            // 1.创建连接
            Connection rabbitMQConnections = RabbitMQConnectionFactory.getRabbitMQConnections();
            // 2.获取通道
            Channel channel = rabbitMQConnections.createChannel();
            // 3.声明交换机
            channel.exchangeDeclare(topic_EXCHANGE_NAME, "topic");
            // 4.生命队列
            channel.queueDeclare(topic_QUEUE_NUM_ONE, false, false, false, null);
            // 交换机和队列匹配在一起
            channel.queueBind(topic_QUEUE_NUM_ONE, topic_EXCHANGE_NAME, "*.orange.*");
            DeliverCallback deliverCallback = new DeliverCallback() {
                @Override
                public void handle(String consumerTag, Delivery delivery) throws IOException {
                    String message = new String(delivery.getBody(), "UTF-8");
                    StringBuilder builder = new StringBuilder();
                    builder.append(new Date()).append("---").append("消费者1接收到的消息==========>").append(message);
                    System.out.println(builder.toString());
                    channel.basicAck(delivery.getEnvelope().getDeliveryTag(), false);
                }
            };
            // 4.设置手动应答
            channel.basicConsume(topic_QUEUE_NUM_ONE, false, deliverCallback, consumerTag -> {
            });
        }
    }

    消费者2

    public class RabbitMQtopicReceiver2 {
        // 创建一个topic交换机
        private static final String topic_EXCHANGE_NAME = "topic_exchange_name";
        // 创建队列一
        private static final String topic_QUEUE_NUM_TWO = "topic_queue_num_two";
    
        public static void main(String[] args) throws IOException, TimeoutException {
            // 1.创建连接
            Connection rabbitMQConnections = RabbitMQConnectionFactory.getRabbitMQConnections();
            // 2.获取通道
            Channel channel = rabbitMQConnections.createChannel();
            // 3.声明交换机
            channel.exchangeDeclare(topic_EXCHANGE_NAME, "topic");
            // 声明队列
            channel.queueDeclare(topic_QUEUE_NUM_TWO, false, false, false, null);
            // 交换机和队列匹配在一起
            channel.queueBind(topic_QUEUE_NUM_TWO, topic_EXCHANGE_NAME, "*.*.rabbit");
            channel.queueBind(topic_QUEUE_NUM_TWO, topic_EXCHANGE_NAME, "lazy.#");
            DeliverCallback deliverCallback = new DeliverCallback() {
                @Override
                public void handle(String consumerTag, Delivery delivery) throws IOException {
                    String message = new String(delivery.getBody(), "UTF-8");
                    StringBuilder builder = new StringBuilder();
                    builder.append(new Date()).append("---").append("消费者2接收到的消息==========>").append(message);
                    System.out.println(builder.toString());
                    channel.basicAck(delivery.getEnvelope().getDeliveryTag(), false);
                }
            };
            // 4.设置手动应答
            channel.basicConsume(topic_QUEUE_NUM_TWO, false, deliverCallback, consumerTag -> {
            });
        }
    }

    执行结果如下:

    图三是消费者1接收到的消息,图四是消费者2收到的消息

                                      图三

                                        图四

    从运行结果可以看出。交换机中的消息,根据routingkey模糊匹配,进行分发到了不同的队列。

      

  • 相关阅读:
    Coolite comboBox控件动态显示树形结构(无限树)
    WinXP 允许搜索PHP格式
    LINQ To DataSet 几个常用示例
    在Windows系统上安装PHP工作环境
    将LINQ To DataSet 传回之对象集转成DataTable的函数
    .net wsdl文件用法
    上班了,抱怨一下
    写在情人节
    快乐云南行
    单车骑天下 VS 公益旅游活动
  • 原文地址:https://www.cnblogs.com/yingxiaocao/p/13335710.html
Copyright © 2020-2023  润新知