返回顶部
首页 > 资讯 > 后端开发 > JAVA >【RabbitMQ】常用消息模型详解
  • 446
分享到

【RabbitMQ】常用消息模型详解

java-rabbitmqrabbitmqjava 2023-08-17 10:08:44 446人浏览 安东尼
摘要

文章目录 AMQP协议的回顾RabbitMQ支持的消息模型第一种模型(直连)开发生产者开发消费者生产者、消费者开发优化API参数细节 第二种模型(work quene)开发生产者开发消费者消息自动确认机制 第三种模型(fan

AMQP协议的回顾

在这里插入图片描述
RabbitMQ中有生产者、消费者的概念,生产者先与我们的RabbitMQServer建立连接,建立完连接之后,它会把消息通过连接中通道的形式去传递我们的消息。每一个生产者会对应一个专门的虚拟主机。

在我们做项目的时候,RabbitMQ希望我们每一个项目具有单独的虚拟主机,这样我们多个应用在操作同一个RabbitMQServer的时候互不影响,所以这里的虚拟机有点像关系型数据库中的库概念。

我们在访问虚拟主机的时候是需要权限的,如果需要访问到某一个具体的虚拟主机,我们需要将虚拟主机与用户进行绑定。

比如RabbitMQ默认为我们提供的guest账户,他是可以访问所有的虚拟主机的,具有至高无上的权限。在我们实际的生产环境中我们一般是一个项目访问一个虚拟主机,或者说是一个业务访问一个虚拟主机,在访问的时候我们一般为一个虚拟主机绑定特定的用户。

当我们的生产者通过通道将消息放入到虚拟机之中,因为RabbitMQ存在许多的消息模型,所以这里不一定会把消息放入到交换机之中。也就是说当生产者将消息传递给交换机或者队列之后,他的任务就告一段落了。

这个时候我们的生产者和消费者是完全解耦的,我们不需要关心生产者到底有没有运行,我只关心消费者监听的队列里面有没有对应的消息即可。

消费者在消费消息的时候也需要去连接到我们RabbitMQServer以及虚拟主机,我们才能消费到对应主机中的消息队列里面的数据。

RabbitMQ支持的消息模型

在这里插入图片描述
在这里插入图片描述

最新的版本有第七种消息模型:消息确认模型

第一种模型(直连)

在这里插入图片描述

在上图的模型中,有以下概念:

  • P:生产者,也就是要发送消息的程序
  • C:消费者:消息的接受者,会一直等待消息到来。
  • queue:消息队列,图中红色部分。类似一个邮箱,可以缓存消息;生产者向其中投递消息,消费者从其中取出消息。

首先我们先创建一个新用户/ems,然后将一个虚拟主机与其绑定,然后给他添加超级用户权限:
在这里插入图片描述

注意:
用户名必须以/开头

开发生产者

public class Provider {    //生产消息    @Test    public void testSendMessage() throws ioException, TimeoutException {        //创建连接mq的连接工厂对象        ConnectionFactory connectionFactory = new ConnectionFactory();        //设置连接rabbitmqserver主机        connectionFactory.setHost("10.15.0.9");        //设置端口号        connectionFactory.setPort(5672);        //设置连接那个虚拟主机        connectionFactory.setVirtualHost("/ems");        //设置访问虚拟主机的用户名和密码        connectionFactory.setUsername("ems");        connectionFactory.setPassword("123");        //获取连接对象        Connection connection = connectionFactory.newConnection();              //获取连接中通道        Channel channel = connection.createChannel();        //通道绑定对应消息队列        //参数1:  队列名称 如果队列不存在自动创建        //参数2:  用来定义队列特性是否要持久化 true 持久化队列   false 不持久化        //参数3:  exclusive 是否独占队列  true 独占队列   false  不独占        //参数4:  autoDelete: 是否在消费完成后自动删除队列  true 自动删除  false 不自动删除        //参数5:  额外附加参数        channel.queueDeclare("hello",true,false,false,null);        //发布消息        //参数1: 交换机名称 参数2:队列名称  参数3:传递消息额外设置  参数4:消息的具体内容        channel.basicPublish("","hello", null,"hello rabbitmq".getBytes());channel.close();connection.close();    }}

开发消费者

public class Customer {    public static void main(String[] args) throws IOException, TimeoutException {        //创建连接工厂        ConnectionFactory connectionFactory = new ConnectionFactory();        connectionFactory.setHost("10.15.0.9");        connectionFactory.setPort(5672);        connectionFactory.setVirtualHost("/ems");        connectionFactory.setUsername("ems");        connectionFactory.setPassword("123");        //创建连接对象        Connection connection = connectionFactory.newConnection();*/               //创建通道        Channel channel = connection.createChannel();        //通道绑定对象        channel.queueDeclare("hello",true,false,false,null);        //消费消息        //参数1: 消费那个队列的消息 队列名称        //参数2: 开始消息的自动确认机制        //参数3: 消费时的回调接口        channel.basicConsume("hello",true,new DefaultConsumer(channel){            @Override //最后一个参数: 消息队列中取出的消息            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {                System.out.println("===================================="+new String(body));            }        });    }}

注意:
在使用Junit测试的时候,他是不支持多线程模型的。如果我们使用@Test去运行的话,他没法让我们的消费者去监听(运行完之后直接就杀死了该进程,不会处于监听状态),所以这里我们要换成一个main函数。
生产者则不需要注意这一点,因为它生产完消息就完事了

在这里插入图片描述
我们发现生产者生产完消息之后,会关闭通道和链接,而在消费这里我们并没有这么做。这是因为可能会导致我们的回调函数还没来得及执行,我们的通道就已经关闭。

该模型的特点:
点对点的简单消费模型。
适用于登录、注册场景

生产者、消费者开发优化

我们发现我们在开发生产者、消费者的时候前面的连接部分代码重复冗余,所以我们可以使用一个工具类对其进行封装:

public class RabbitMQUtils {    private static ConnectionFactory connectionFactory;    private static Properties properties;    static{        //重量级资源  类加载执行之执行一次        connectionFactory = new ConnectionFactory();        connectionFactory.setHost("10.15.0.5");        connectionFactory.setPort(5672);        connectionFactory.setVirtualHost("/");        connectionFactory.setUsername("guest");        connectionFactory.setPassword("guest");    }    //定义提供连接对象的方法    public static Connection getConnection() {        try {            return connectionFactory.newConnection();        } catch (Exception e) {            e.printStackTrace();        }        return null;    }    //关闭通道和关闭连接工具方法    public static void closeConnectionAndChanel(Channel channel, Connection conn) {        try {            if(channel!=null) channel.close();            if(conn!=null)   conn.close();        } catch (Exception e) {            e.printStackTrace();        }    }}

我们这里使用静态代码块是因为connectionFactory是重量级资源,所以我们决定只在类加载执行时执行一次。

我们这里稍微复习一下java的静态代码块,我们会发现在一些项目源码中经常会见到他。
静态代码块语法格式:

static{}

静态代码块的特点:随着类的加载而执行,而且只执行一次
执行优先级高于非静态的初始化块,它会在类初始化的时候执行一次,执行完成便销毁,它仅能初始化类变量,即static修饰的数据成员。
那么正好我们再来提一下非静态代码块:
非静态代码块语法格式:

{}

执行的时候如果有静态初始化块,先执行静态初始化块再执行非静态初始化块,在每个对象生成时都会被执行一次,它可以初始化类的实例变量。非静态初始化块会在构造函数执行时,在构造函数主体代码执行之前被运行。
执行顺序:
静态代码块----->非静态代码块-------->构造函数

api参数细节

生产者和消费者均有一个方法queueDeclare,就是声明操作的队列:

channel.queueDeclare("hello",true,false,false,null);
  • 参数1: 队列名称
    • 如果队列不存在自动创建
  • 参数2: 用来定义队列特性是否要持久化
    • true 持久化队列
    • false 不持久化
    • 注意这里说的是队列的持久化
    • 也就是如果开启持久化的话,我们即使重启rabbitmq服务该队列也会存在,因为其内部会把队列从内存写到硬盘中去。当重启完成之后,其又会重新将硬盘中的队列读到内存中去
  • 参数3: exclusive 是否独占队列
    • true 独占队列
    • 也就是说队列只能被当前通道所绑定
    • false 不独占
  • 参数4: autoDelete: 是否在消费完成后自动删除队列
    • true 自动删除
    • false 不自动删除
    • 这里的自动删除队列是指消费者不再监听占用队列,队列才会消失
  • 参数5: 额外附加参数

注意:直连模型下,消费者和生产者的queueDeclare中的参数要保持一致,这样才能保证操作的是同一个队列

生产者:

channel.basicPublish("","hello", MessageProperties.PERSISTENT_TEXT_PLAIN,"hello rabbitmq".getBytes());
  • 参数1: 交换机名称 (我们这里没有使用交换机所以没有指定)
  • 参数2:队列名称
  • 参数3:传递消息额外设置
    • MessageProperties.PERSISTENT_TEXT_PLAIN
    • 我们可以通过此参数设置消息在队列中的持久化
  • 参数4:消息的具体内容
    • 这里是以字节的方式进行传输

消费者:

channel.basicConsume("hello",true,new DefaultConsumer(channel){      @Override //最后一个参数: 消息队列中取出的消息      public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {         System.out.println("======"+new String(body));      }});
  • 参数1: 消费哪个队列的消息 队列名称
  • 参数2: 开始消息的自动确认机制
  • 参数3: 消费时的回调接口
    • 这里我们可以传入一个consumer对象,而这个consumer是一个接口,它有一个实现类DefaultConsumer

第二种模型(work quene)

Work queues,也被称为(Task queues),任务模型。当消息处理比较耗时的时候,可能生产消息的速度会远远大于消息的消费速度。长此以往,消息就会堆积越来越多,无法及时处理。此时就可以使用work 模型:让多个消费者绑定到一个队列,共同消费队列中的消息队列中的消息一旦消费,就会消失,因此任务是不会被重复执行的。

在这里插入图片描述

角色:

  • P:生产者:任务的发布者
  • C1:消费者-1,领取任务并且完成任务,假设完成速度较慢
  • C2:消费者-2:领取任务并完成任务,假设完成速度快

开发生产者

public class Provider {    public static void main(String[] args) throws IOException {        //获取连接对象        Connection connection = RabbitMQUtils.getConnection();        //获取通道对象        Channel channel = connection.createChannel();        //通过通道声明队列        channel.queueDeclare("work", true, false, false, null);        for (int i = 1; i <=20; i++) {            //生产消息            channel.basicPublish("", "work", null, (i + "hello work quene").getBytes());        }        //关闭资源        RabbitMQUtils.closeConnectionAndChanel(channel, connection);    }}

我们这里使用了前面提到的连接工具类

我们运行我们的代码:
在这里插入图片描述
这里的Unacked代表未被确认的消息

开发消费者

如果我们对两个消费者不做任何处理:

消费者-1

public class Customer1 {    public static void main(String[] args) throws IOException {        //获取连接        Connection connection = RabbitMQUtils.getConnection();        final Channel channel = connection.createChannel();                channel.queueDeclare("work",true,false,false,null);        //参数1:队列名称  参数2:消息自动确认 true  消费者自动向rabbitmq确认消息消费  false 不会自动确认消息        channel.basicConsume("work",false,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));}        });    }}

消费者-2

public class Customer2 {    public static void main(String[] args) throws IOException {        //获取连接        Connection connection = RabbitMQUtils.getConnection();        final Channel channel = connection.createChannel();                channel.queueDeclare("work",true,false,false,null);        //参数1:队列名称  参数2:消息自动确认 true  消费者自动向rabbitmq确认消息消费  false 不会自动确认消息        channel.basicConsume("work",false,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));}        });    }}

在这种不做任何处理的情况下,消费者1、消费者2消费的消息都是一致的:
在这里插入图片描述
在这里插入图片描述

总结:默认情况下,RabbitMQ将按顺序将每个消息发送给下一个使用者。平均而言,每个消费者都会收到相同数量的消息。这种分发消息的方式称为循环。(也就是说平均分配)

而这样的话我们不难想到一个问题:加入我们的消费者1处理的比较慢,消费者2处理的比较快。这就导致消费者1的消息会在队列中造成滞留,消费者2可能已经处理完闲着了。这样的情况下平均分配显然也会影响效率,并且导致消息再队列中的积累。

我们可以模拟一下这个情况,我们在消费者1中添加一个线程睡眠:

在这里插入图片描述

这个时候我们运行发现,在消费者2将自己的消息打印完之后,消费者1的消息只打印了一条:
在这里插入图片描述

在这里插入图片描述

那么能不能用第二种模型实现一种能者多劳的模式呢?

消息自动确认机制

Doing a task can take a few seconds. You may wonder what happens if one of the consumers starts a long task and dies with it only partly done. With our current code, once RabbitMQ delivers a message to the consumer it immediately marks it for deletion. In this case, if you kill a worker we will lose the message it was just processing. We’ll also lose all the messages that were dispatched to this particular worker but were not yet handled.

But we don’t want to lose any tasks. If a worker dies, we’d like the task to be delivered to another worker.
完成一项任务可能需要几秒钟。你可能想知道,如果其中一个消费者开始了一项长任务,但只完成了一部分任务就去世了,会发生什么。使用我们当前的代码,一旦RabbitMQ将消息传递给消费者,它会立即将其标记为删除。在这种情况下,如果你杀死一个消费者,我们将丢失它正在处理的消息。我们还将丢失发送给该特定工作人员但尚未处理的所有消息。
但我们不想失去任何任务。如果一名消费者死亡,我们希望将任务交付给另一名消费者。

自动确认是指,当消息一旦被Consumer接收到,则自动确认收到,并将相应 message 从 RabbitMQ 的消息缓存中移除。但是在实际业务处理中,很可能消息接收到,业务处理出现异常,那么该消息就会丢失。如果设置了手动确认方式,则需要在业务处理成功后,调用channel.basicAck(),手动签收,如果出现异常,则调用channel.basicNack()方法,让其自动重新发送消息。

在这里插入图片描述

我们使用能者多劳的模式需要进行两步额外的操作:

  • 设置通道一次只能消费一个消息

  • 关闭消息的自动确认,开启手动确认消息

Customer1:

public class Customer1 {    public static void main(String[] args) throws IOException {        //获取连接        Connection connection = RabbitMQUtils.getConnection();        final Channel channel = connection.createChannel();        channel.basicQos(1);//每一次只能消费一个消息        channel.queueDeclare("work",true,false,false,null);        //参数1:队列名称  参数2:消息自动确认 true  消费者自动向rabbitmq确认消息消费  false 不会自动确认消息        channel.basicConsume("work",false,new DefaultConsumer(channel){            @Override            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {                try{                    Thread.sleep(2000);                }catch (Exception e){                    e.printStackTrace();                }                System.out.println("消费者-1: "+new String(body));                // 参数1:确认队列中那个具体消息 参数2:是否开启多个消息同时确实                channel.basicAck(envelope.getDeliveryTag(),false);            }        });    }}

Customer2:

public class Customer2 {    public static void main(String[] args) throws IOException {        //获取连接        Connection connection = RabbitMQUtils.getConnection();        final Channel channel = connection.createChannel();        channel.basicQos(1);        channel.queueDeclare("work",true,false,false,null);        channel.basicConsume("work",false,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));                //手动确认  参数1:手动确认消息标识  参数2:false 每次确认一个                channel.basicAck(envelope.getDeliveryTag(), false);            }        });        }}

我们对上面两段代码的一些参数或方法做出解释:

basicQos()

void basicQos(int prefetchSize, int prefetchCount, boolean global) throws IOException;

参数:

  • prefetchSize:消息的大小

  • prefetchCount:会告诉RabbitMQ不要同时给一个消费者推送多于N个消息,即一旦有N个消息还没有ack,则该consumer将block掉,直到有消息ack

  • global:是否将上面设置应用于channel,简单点说,就是上面限制是channel级别的还是consumer级别

channel.basicAck()

void basicAck(long deliveryTag, boolean multiple) throws IOException;

参数:

  • deliveryTag:该消息的index

  • multiple:是否批量处理.

    • true:将一次性ack所有小于deliveryTag的消息

envelope.getDeliveryTag()

在这里插入图片描述
这个方法是表示消息的唯一标识ID,返回的是一个正整数,是rabbitmq来自增设置的

第三种模型(fanout)

fanout 扇出 也称为广播

在这里插入图片描述

在广播模式下,消息发送流程是这样的:

  • 可以有多个消费者
  • 每个消费者有自己的queue(队列)
    • 我们这里创建的队列是临时的,用完之后就会删除
  • 每个队列都要绑定到Exchange(交换机)
  • 生产者发送的消息,只能发送到交换机,交换机来决定要发给哪个队列,生产者无法决定。
  • 交换机把消息发送给绑定过的所有队列
  • 队列的消费者都能拿到消息。实现一条消息被多个消费者消费

使用场景:
比如我们的商品购物车,在我们结算的时候,我们可能会跟多个系统进行交互,订单系统、库存系统等等。这个时候我们的购物车信息会被多条队列给消费。

在fanout模式下,路由的相关配置没有意义,相关参数可以空着

开发生产者

public class Provider {    public static void main(String[] args) throws IOException {        //获取连接对象        Connection connection = RabbitMQUtils.getConnection();        Channel channel = connection.createChannel();        //将通道声明指定交换机   //参数1: 交换机名称    参数2: 交换机类型  fanout 广播类型        channel.exchangeDeclare("logs","fanout");        //发送消息        channel.basicPublish("logs","",null,"fanout type message".getBytes());        //释放资源        RabbitMQUtils.closeConnectionAndChanel(channel,connection);    }}

channel.exchangeDeclare("logs","fanout");:

  • 参数1: 交换机名称
  • 参数2: 交换机类型
    • fanout是广播类型

开发消费者

public class Customer1 {    public static void main(String[] args) throws IOException {        //获取连接对象        Connection connection = RabbitMQUtils.getConnection();        Channel channel = connection.createChannel();        //通道绑定交换机        channel.exchangeDeclare("logs","fanout");        //临时队列        String queueName = channel.queueDeclare().getQueue();        //绑定交换机和队列        channel.queueBind(queueName,"logs","");        //消费消息        channel.basicConsume(queueName,true,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));            }        });    }}

其他几个消费者同理

我们创建三个消费者,把他们开启之后,再开启我们的生产者,测试结果如下:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

第四种模型(Routing)

在这里插入图片描述
第五种模型其实是第四种模型的一个分支,如果我们叫第四种模型为路由的话,那么我们可以说第五种模型是动态路由。第四种模型我们也可以叫做direct模型(直连)

在Fanout模式中,一条消息,会被所有订阅的队列都消费。但是,在某些场景下,我们希望不同的消息被不同的队列消费。这时就要用direct类型的Exchange。

在Routing模型下:

  • 队列与交换机的绑定,不能是任意绑定了,而是要指定一个RoutingKey(路由key)
  • 消息的发送方在 向 Exchange发送消息时,也必须指定消息的 RoutingKey
  • Exchange不再把消息交给每一个绑定的队列,而是根据消息的Routing Key进行判断,只有队列的Routingkey与消息的 Routing key完全一致,才会接收到消息

流程:

在这里插入图片描述

图解:

  • P:生产者,向Exchange发送消息,发送消息时,会指定一个routing key。
  • X:Exchange(交换机),接收生产者的消息,然后把消息递交给 与routing key完全匹配的队列
  • C1:消费者,其所在队列指定了需要routing key 为 error 的消息
  • C2:消费者,其所在队列指定了需要routing key 为 info、error、warning 的消息

开发生产者

public class Provider {    public static void main(String[] args) throws IOException {        //获取连接对象        Connection connection = RabbitMQUtils.getConnection();        //获取连接通道对象        Channel channel = connection.createChannel();        String exchangeName = "logs_direct";        //通过通道声明交换机  参数1:交换机名称  参数2:direct  路由模式        channel.exchangeDeclare(exchangeName,"direct");        //发送消息        String routingkey = "error";        channel.basicPublish(exchangeName,routingkey,null,("指定的route key"+key+"的消息").getBytes());        //关闭资源        RabbitMQUtils.closeConnectionAndChanel(channel,connection);    }}

这个时候我们就已经可以看到我们的交换机了:
在这里插入图片描述

开发消费者

我们这里开发两个消费者:

  • 消费者1拿到routekey为error的消息
  • 消费者2拿到routekey为info、error、warning的消息
public class Customer1 {    public static void main(String[] args) throws IOException {        Connection connection = RabbitMQUtils.getConnection();        Channel channel = connection.createChannel();        String exchangeName = "logs_direct";        //通道声明交换机以及交换的类型        channel.exchangeDeclare(exchangeName,"direct");        //创建一个临时队列        String queue = channel.queueDeclare().getQueue();        //基于route key绑定队列和交换机        channel.queueBind(queue,exchangeName,"error");        //获取消费的消息        channel.basicConsume(queue,true,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));            }        });    }}
public class Customer2 {    public static void main(String[] args) throws IOException {        Connection connection = RabbitMQUtils.getConnection();        Channel channel = connection.createChannel();        String exchangeName = "logs_direct";        //声明交换机 以及交换机类型 direct        channel.exchangeDeclare(exchangeName,"direct");        //创建一个临时队列        String queue = channel.queueDeclare().getQueue();        //临时队列和交换机绑定        channel.queueBind(queue,exchangeName,"info");        channel.queueBind(queue,exchangeName,"error");        channel.queueBind(queue,exchangeName,"warning");        //消费消息        channel.basicConsume(queue,true,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));            }        });    }}

我们测试一下:

测试生产者发送Route key为error的消息时

在这里插入图片描述

在这里插入图片描述

测试生产者发送Route key为info的消息时

在这里插入图片描述

在这里插入图片描述


第五种模型(Topic)

Topic类型的ExchangeDirect相比,都是可以根据RoutingKey把消息路由到不同的队列。只不过Topic类型Exchange可以让队列在绑定Routing key 的时候使用通配符!这种模型Routingkey 一般都是由一个或多个单词组成,多个单词之间以”.”分割,例如: item.insert

在这里插入图片描述

  • *(star) :匹配不多不少恰好1个词
  • #(hash): 匹配0个或多个词

例如:

audit.#    匹配audit.irs.corporate或者 audit.irs 等audit.*   只能匹配audit.irs

开发生产者

public class Provider {    public static void main(String[] args) throws IOException {        //获取连接对象        Connection connection = RabbitMQUtils.getConnection();        Channel channel = connection.createChannel();        //声明交换机以及交换机类型 topic        channel.exchangeDeclare("topics","topic");        //发布消息        String routekey = "user";        channel.basicPublish("topics",routekey,null,("这里是topic动态路由模型,routekey: ["+routekey+"]").getBytes());        //关闭资源        RabbitMQUtils.closeConnectionAndChanel(channel,connection);    }}

开发消费者

我们还是开发两个消费者:

消费者1Routing Key中使用*通配符方式

public class Customer1 {    public static void main(String[] args) throws IOException {        //获取连接        Connection connection = RabbitMQUtils.getConnection();        Channel channel = connection.createChannel();        //声明交换机以及交换机类型        channel.exchangeDeclare("topics","topic");        //创建一个临时队列        String queue = channel.queueDeclare().getQueue();        //绑定队列和交换机  动态统配符形式route key        channel.queueBind(queue,"topics","user.*");        //消费消息        channel.basicConsume(queue,true,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));            }        });    }}

消费者2中Routing Key中使用#通配符方式

public class Customer2 {    public static void main(String[] args) throws IOException {        //获取连接        Connection connection = RabbitMQUtils.getConnection();        Channel channel = connection.createChannel();        //声明交换机以及交换机类型        channel.exchangeDeclare("topics","topic");        //创建一个临时队列        String queue = channel.queueDeclare().getQueue();        //绑定队列和交换机  动态统配符形式route key        channel.queueBind(queue,"topics","user.#");        //消费消息        channel.basicConsume(queue,true,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));            }        });    }}

这个时候我们测试的结果就是:

  • 消费者2可以拿到消息
  • 消费者1拿不到消息

来源地址:https://blog.csdn.net/zyb18507175502/article/details/127561131

--结束END--

本文标题: 【RabbitMQ】常用消息模型详解

本文链接: https://lsjlt.com/news/372658.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 【RabbitMQ】常用消息模型详解
    文章目录 AMQP协议的回顾RabbitMQ支持的消息模型第一种模型(直连)开发生产者开发消费者生产者、消费者开发优化API参数细节 第二种模型(work quene)开发生产者开发消费者消息自动确认机制 第三种模型(fan...
    99+
    2023-08-17
    java-rabbitmq rabbitmq java
  • rabbitmq的消息模型有哪些
    RabbitMQ的消息模型有以下几种:1. 点对点模型(Point-to-Point Model):也称为队列模型,消息的发送者(P...
    99+
    2023-09-20
    rabbitmq
  • Java RabbitMQ消息队列详解常见问题
    目录消息堆积保证消息不丢失死信队列延迟队列RabbitMQ消息幂等问题RabbitMQ消息自动重试机制合理的选择重试机制消费者开启手动ack模式rabbitMQ如何解决消息幂等问题R...
    99+
    2024-04-02
  • rabbitmq的五种消息模型是什么
    RabbitMQ支持以下五种消息模型: 简单模式(Simple Mode):生产者将消息发送到队列,消费者从队列中获取消息并处理...
    99+
    2023-10-25
    rabbitmq
  • springboot-rabbitmq-reply 消息直接回复模式详情
    目录一、使用场景二、Reply实战(1)依赖与YML配置(2)RabbitMq bean配置(3)消息生产端(4)消息消费端(1)方法一:sendTo注解+方法返回值(2)方法二:读...
    99+
    2024-04-02
  • 消息交换模式RabbitMQ简介
    RabbitMQ是AMQP的一个典型实现,它消息发布者的消息发布到Exchange上,同时需要制定routingkey,可以通过指定交换机的不同模式实现不同的行为。 RabbitMQ...
    99+
    2024-04-02
  • 详解消息队列及RabbitMQ部署和使用
    目录什么是消息队列为什么需要消息队列常见的消息队列ActiveMQRabbitMQZeroMQKafkaRocketMQRabbitMQ 的部署和使用Python 编写生产者Pyth...
    99+
    2024-04-02
  • Android 消息队列模型详解及实例
    Android 消息队列模型详解及实例Android系统的消息队列和消息循环都是针对具体线程的,一个线程可以存在(当然也可以不存在)一个消息队列(Message Queue)和一个消息循环(Looper)。Android中除了UI线程(主线...
    99+
    2023-05-31
    android 消息队列 roi
  • SpringBoot+RabbitMQ实现消息可靠传输详解
    目录环境配置消息丢失分析生产阶段生产端模拟消息丢失RabbitMQ消费端环境配置 SpringBoot 整合 RabbitMQ 实现消息的发送。 1.添...
    99+
    2024-04-02
  • Kafka Producer中的消息缓存模型图解详解
    目录前言什么是消息累加器RecordAccumulator消息缓存模型ProducerBatch的内存大小内存分配Batch的创建和释放问题和答案总结前言 在阅读本文之前, 希望你可...
    99+
    2024-04-02
  • rabbitmq消息阻塞怎么解决
    RabbitMQ消息阻塞的原因可能是由于消费者消费速度慢于生产者生产速度导致的,解决方法可以有以下几种:1. 增加消费者数量:可以通...
    99+
    2023-09-20
    rabbitmq
  • rabbitmq消息积压怎么解决
    要解决RabbitMQ消息积压问题,可以采取以下几个方法:1. 增加消费者:可以增加消费者的数量来提高消息的处理速度,从而减少消息的...
    99+
    2023-09-20
    rabbitmq
  • 解决RabbitMq消息队列Qos Prefetch消息堵塞问题
    mq是实现代码扩展的有利手段,个人喜欢用概念来学习新知识,介绍堵塞问题的之前,先来段概念的学习。 ConnectionFactory:创建connection的工厂类 Connect...
    99+
    2024-04-02
  • springboot中rabbitmq实现消息可靠性机制详解
    1. 生产者模块通过publisher confirm机制实现消息可靠性  1.1 生产者模块导入rabbitmq相关依赖 <!--AMQP依赖,包含Rabbit...
    99+
    2024-04-02
  • 详解SpringBoot整合RabbitMQ如何实现消息确认
    目录简介生产者消息确认介绍流程配置ConfirmCallbackReturnCallback注册ConfirmCallback和ReturnCallback消费者消息确认介绍手动确认...
    99+
    2024-04-02
  • 详解RabbitMq如何做到消息的可靠性投递
    目录前言RabbitMq的投递及消费流程提供者如何确保消息的成功投递单条消息的同步确认多条消息的同步确认异步消息确认消息的返回机制前言 现在的一些互联网项目或者是高并发的项目中很少有...
    99+
    2024-04-02
  • 如何解决RabbitMq消息队列Qos Prefetch消息堵塞问题
    本篇内容介绍了“如何解决RabbitMq消息队列Qos Prefetch消息堵塞问题”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!...
    99+
    2023-06-29
  • springBoot整合rabbitmq测试常用模型小结
    目录1.添加依赖2.编写配置3.编写并测试之前我们记录了原生java代码使用rabbitmq的方法,很简单,类似于原生jdbc代码一样,将连接对象抽离出来作为工具类,生产者和消费者通...
    99+
    2024-04-02
  • 利用Python学习RabbitMQ消息队列
    RabbitMQ可以当做一个消息代理,它的核心原理非常简单:即接收和发送消息,可以把它想象成一个邮局:我们把信件放入邮箱,邮递员就会把信件投递到你的收件人处,RabbitMQ就是一个邮箱、邮局、投递员功能综...
    99+
    2022-06-04
    队列 消息 Python
  • RabbitMQ消息转换器怎么应用
    本文小编为大家详细介绍“RabbitMQ消息转换器怎么应用”,内容详细,步骤清晰,细节处理妥当,希望这篇“RabbitMQ消息转换器怎么应用”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。在SpringAMQP的发...
    99+
    2023-07-05
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作