订阅模式 Publish/Subscribe

模型图

springboot RabbitMq多个消费者 rabbitmq一条消息多个消费者_System

我们之前学习的都是一个消息只能被一个消费者消费,那么如果我想发一个消息 能被多个消费者消费,这时候怎么
办? 这时候我们就得用到了消息中的发布订阅模型
在前面的教程中,我们创建了一个工作队列,都是一个任务只交给一个消费者。
这次我们做 将消息发送给多个消费者。这种模式叫做“发布/订阅”。
举列:
类似微信订阅号 发布文章消息 就可以广播给所有的接收者。(订阅者)
那么咱们来看一下图,我们学过前两种有一些不一样,work 模式 是不是同一个队列 多个消费者,而 ps 这种模式呢,是
一个队列对应一个消费者,pb 模式还多了一个 X(交换机 转发器) ,这时候我们要获取消息 就需要队列绑定到交换机
上,交换机把消息发送到队列 , 消费者才能获取队列的消息
解读:
1、1 个生产者,多个消费者
2、每一个消费者都有自己的一个队列
3、生产者没有将消息直接发送到队列,而是发送到了交换机(转发器)
4、每个队列都要绑定到交换机
5、生产者发送的消息,经过交换机,到达队列,实现,一个消息被多个消费者获取的目的

因为交换机没有存储消息的能力,在 rabbitmq 中只有队列存储消息的
能力.因为这时还没有队列,所以就会丢失;
小结:消息发送到了一个没有绑定队列的交换机时,消息就会丢失

这里注意一点:在订阅模式是有一个新的概念就是交换器正如模型图一样,与之前的轮询和公平分发模式不一样了,之后的消费者中交换机必须与生产者绑定的交换机是同一个(自己在开始写的时候犯过这样的错误)

public class Send {

private final static String EXCHANGE_NAME = "test_exchange_topic";

public static void main(String[] args) throws IOException, TimeoutException {
    //链接MQ,并且创建channel
    Connection connection = ConnectionUtils.getConnection();
    Channel channel = connection.createChannel();

    //声明echange交换机
    channel.exchangeDeclare(EXCHANGE_NAME, "topic");

    //消息内容
    String message = "id=1001";
    channel.basicPublish(EXCHANGE_NAME, "item.add", null, message.getBytes());
    System.out.println(" [x] Sent '" + message + "'");
    channel.close();
    connection.close();
 }
}

public class Recv {
private final static String EXCHANGE_NAME = "test_exchange_topic";

private final static String QUEUE_NAME = "test_queue_topic_1";

public static void main(String[] args) throws IOException, TimeoutException {
    //链接MQ 并创建channel
    Connection connection = ConnectionUtils.getConnection();
    final Channel channel = connection.createChannel();

    //声明队列
    channel.queueDeclare(QUEUE_NAME, false, false, false, null);

    //绑定交换机
    channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "item.update");
    channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "item.add");

    //同时有一个服务器会发消息给消费者
    channel.basicQos(1);

    //定义消费者
    Consumer consumer = new DefaultConsumer(channel) {
        @Override
        public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
            String msg = new String(body, "utf-8");
            System.out.println("[1] Recv msg:" + msg);
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                System.out.println("[1] done ");
                //手动回执
                channel.basicAck(envelope.getDeliveryTag(), false);
            }
        }
    };
    boolean autoAck = false;
    channel.basicConsume(QUEUE_NAME, autoAck, consumer);

}
}

public class Recv1 {

    private final static String EXCHANGE_NAME = "test_exchange_topic";

    private final static String QUEUE_NAME = "test_queue_topic_2";

    public static void main(String[] args) throws IOException, TimeoutException {
        //链接MQ 并创建channel
        Connection connection = ConnectionUtils.getConnection();
        final Channel channel = connection.createChannel();

        //声明队列
        channel.queueDeclare(QUEUE_NAME, false, false, false, null);

        //绑定交换机
        channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "item.#");

        //同时有一个服务器会发消息给消费者
        channel.basicQos(1);

        //定义消费者
        Consumer consumer = new DefaultConsumer(channel) {
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                String msg = new String(body, "utf-8");
                System.out.println("[2] Recv msg:" + msg);
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {
                    System.out.println("[2] done ");
                    //手动回执
                    channel.basicAck(envelope.getDeliveryTag(), false);
                }
            }
        };
        boolean autoAck = false;
        channel.basicConsume(QUEUE_NAME, autoAck, consumer);

    }
}

springboot RabbitMq多个消费者 rabbitmq一条消息多个消费者_消息发送_02