简单入门RabbitMQ

本章将带大家来写一个简单的程序,使用 Java 创建RabbitMQ 的生产者和消费者

依赖引入

在 Maven 仓库中输入 amqp-client:

找到第一个 RabbitMQ Java Client ,点击进去找到一个合适的版本然后将依赖引入到我们项目中的 pom.xml 文件中。

生产者代码编写

首先我们来回顾 RabbitMQ 的工作流程:

我们首先要建立连接Connection,通过 ConnectionFactory 来创建一个连接,在建立连接前我们需要设置好RabbitMQ 的参数:主机、端口号、用户名、密码、绑定的虚拟机。

java 复制代码
		//建立连接
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost(Constants.HOST);
        factory.setPort(Constants.PORT);
        factory.setUsername(Constants.NAME);
        factory.setPassword(Constants.PASSWORD);
        factory.setVirtualHost(Constants.VIRTUAL_HOST);
        Connection connection = factory.newConnection();

接着我们需要开启信道 Channel,因为我们的Connection 是抽象的虚拟连接,并不是真正和RabbitMQ 虚拟机进行了连接,我们需要Connection 上的信道来进行通信:

java 复制代码
		//开启信道
        Channel channel = connection.createChannel();

声明交换机,当我们没有显式地声明交换机的话,RabbitMQ 会使用默认的交换机,RabbitMQ默认会为每个连接自动创建一个名为 ""(空字符串)的直连交换机(Direct Exchange),所有队列都会隐式绑定到该交换机,绑定的路由键(Routing Key)就是队列名称。


声明队列:

java 复制代码
//声明队列,使用内置的交换机,如果队列不存在会自动帮我们创建
channel.queueDeclare(Constants.HELLO_QUEUE, true, false, true, null);

参数介绍:

java 复制代码
Queue.DeclareOk queueDeclare(String queue,
boolean durable, 
boolean exclusive, 
boolean autoDelete,
Map<String, Object> arguments) throws IOException;

queue: 队列名称
durable: 表示是否需要持久化,true 说明需要持久化,那么我们发送的消息就会进行落盘操作,如果RabbitMQ 重启了也可以找到这个消息,即把消息保存在硬盘中,如果是 false ,表示不需要进行持久化操作,那么我们发送的消息就只会在内存中,如果 RabbitMQ 重启了,消息也就不见了。
exclusive:表示是否独占,true: 表示该队列只对它的连接可见,且连接关闭时队列自动删除。

适用于临时场景(如RPC回调队列),确保队列不被其他连接共享。

注意:若其他连接尝试使用排他队列,会抛异常。
autoDelete:表示是否自动删除,true:当最后一个消费者取消订阅(如断开连接)时,队列自动删除。适用于动态队列(如临时任务队列),无需手动清理。

注意:若队列从未有过消费者,则不会触发删除。
arguments:设置队列的高级参数,在后面的章节中会详细展开。


发送消息:

java 复制代码
for (int i = 0; i < 10; i++) {
    String msg = "hello work queue...." + i;
    channel.basicPublish("",Constants.HELLO_QUEUE, null, msg.getBytes());
}

参数说明:

java 复制代码
void basicPublish(String exchange, 
String routingKey, 
BasicProperties props, 
byte[] body) throws IOException;

exchange:交换机的名称,由于上面我们没有声明交换机,所以这里填 "" 【空字符串】表示默认交换机

routingKey:路由键,也就说消息要发送到哪个队列上,路由键即为标识,在下一章节中会详细介绍。

props:属性配置,这个在后面的章节中也会介绍

body:消息内容


资源释放:

java 复制代码
//6. 资源释放
channel.close();
connection.close();

这里我们先释放信道,然后再释放连接。

注意:释放连接之后,就不能释放信道了!!!因为连接都不存在了,信道也不会存在。

因此你也可以只释放连接。


运行程序之后,我们打开 RabbitMQ 的面板:

会发现我们成功将十条消息发送到队列中

通过输入Messages 数量,然后点击 Get Message(s),就可以查看消息的详细信息了。

消费者代码编写

和生产者类似,都需要先设置好虚拟机的参数,然后创建连接,创建信道:

java 复制代码
//建立连接
ConnectionFactory factory = new ConnectionFactory();
factory.setHost(Constants.HOST);
factory.setPort(Constants.PORT);
factory.setUsername(Constants.NAME);
factory.setPassword(Constants.PASSWORD);
factory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = factory.newConnection();
//开启信道
Channel channel = connection.createChannel();

声明队列:

java 复制代码
//声明队列
channel.queueDeclare(Constants.HELLO_QUEUE, true, false, true, null);

为什么消费者需要声明队列?

因为消费者要消费的前提是有队列可以消费

在分布式系统中,生产者和消费者的代码可能不会在同一台机器上,如果生产者还没有启动,反而消费者先启动了,消费者就会抛出异常,找不到队列

因此为了避免发生上面的异常情况,我们也需要在消费者中声明队列,这样,如果没有队列的话,RabbitMQ 就会帮我们创建队列出来。


接收消息和处理消息:

java 复制代码
DefaultConsumer consumer = new DefaultConsumer(channel){
@Override
public void handleDelivery(String consumerTag, 
Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
        System.out.println("接收到的消息:" + new String(body));
    }
};
channel.basicConsume(Constants.HELLO_QUEUE, true, consumer);

处理消息的主逻辑我们通过 DefaultConsumer 来进行编写,通过重写 handleDelivery 方法,来实现我们需要的逻辑。

参数介绍:

java 复制代码
/**
 * No-op implementation of {@link Consumer#handleDelivery}.
 */
@Override
public void handleDelivery(String consumerTag,
                           Envelope envelope,
                           AMQP.BasicProperties properties,
                           byte[] body)
    throws IOException
{
        // no work to do
}

consumerTag :消费者表示,用于区分消费者,就像我们的身份证一样

envelope:消息的元数据包装对象。

properties:消息的附加属性(元数据)

body:消息内容


最后我们可以不主动释放连接,这样我们就可以看到完整的打印信息了

RabbitMQ 的 Java 客户端底层使用 ExecutorService 管理消费者线程。

这些线程默认是守护线程,因此主线程【main】结束后,它们会被 JVM 强制终止。

由于RabbitMQ 线程和 main 线程属于多线程,因此根据多线程知识,我们可以知道如果主动释放连接的话,main 线程就是立即结束,我们可能就看不到完整的信息了。

相关推荐
星星点点洲3 小时前
【RabbitMQ】消息丢失问题排查与解决
分布式·rabbitmq
chilavert3188 小时前
从RPA项目说说RPC和MQ的使用。
开发语言·qt·rpc·rabbitmq
@小了白了兔18 小时前
RabbitMQ工作流程及使用方法
分布式·rabbitmq
噼里啪啦啦.1 天前
RabbitMQ
分布式·rabbitmq
希忘auto1 天前
详解RabbitMQ工作模式之通配符模式
rabbitmq
yuanlaile2 天前
RabbitMQ高并发秒杀、抢购系统、预约系统底层实现逻辑
分布式·rabbitmq·rabbitmq高并发·rabbitmq项目实战·rabbitmq实战教程
MYBOYER2 天前
Kafka、RabbitMQ、RocketMQ的区别
kafka·rabbitmq·rocketmq
椰椰椰耶2 天前
【RabbitMQ】路由模式和通配符模式的具体实现
分布式·rabbitmq
jstart千语2 天前
【消息队列】RabbitMQ基本认识
java·服务器·分布式·rabbitmq