一:RabbitMQ 应用
RabbitMQ 提供了 7 种工作模式用于消息传递,而我们入门案例中的程序实际上就是其中的一种简单模式,下面我们来介绍一下这 7 种工作模式。
1.1 Simple (简单模式)
名称 | 描述 |
---|---|
P(Producer) | 生产者,负责发送消息的程序。 |
C(Consumer) | 消费者,负责接收消息的程序。 |
Queue | 消息队列,图中黄色背景部分,类似一个邮箱,用于缓存消息;生产者向队列中投递消息,消费者从队列中取出消息。 |
特点 | 一个生产者(P),一个消费者(C),消息只能被消费一次,因此也称为点对点(Point-to-Point)模式,适用于只能被单个消费者处理的场景。 |
1.2 Work Queue (工作队列)
在工作队列(Work Queue)模式中,一个生产者(P)向队列中发送消息,多个消费者(如 C1、C2)会从队列中接收不同的消息,RabbitMQ 会将消息分配给各消费者,确保每条消息只被消费一次,不会重复。该模式适用于集群环境中的异步处理场景,例如 12306 的短信通知服务:当用户订票成功后,订单消息会发送到 RabbitMQ,由短信服务从队列中获取订单信息并发送通知,同时在多个短信服务之间实现任务分配。
1.3 Publish / Subscribe 模式
在订阅模式中,图中的 X 表示交换机(Exchange)。相比简单模式,订阅模式增加了交换机这一角色,消息传递过程也随之发生变化。Exchange 的作用是接收生产者发送的消息,并根据指定的路由规则,将消息分发到一个或多个队列中。在 RabbitMQ 中,生产者不会直接将消息发送到队列,而是通过交换机完成路由。RabbitMQ 提供了四种类型的交换机:fanout、direct、topic 和 headers,每种类型对应不同的路由策略。
需要注意的是,Exchange 只负责消息的转发,不具备存储消息的能力。因此,如果没有队列绑定到 Exchange,或者没有符合路由规则的队列,消息将会被丢失。为避免消息丢失,应确保交换机与至少一个队列正确绑定,消息如果发送成功会被多个消费者同时接收,每个消费者接收的消息都相同。
交换机类型 | 描述 |
---|---|
Fanout | 广播模式:将消息发送给所有绑定到交换机的队列,适用于发布/订阅模式。 |
Direct | 定向模式:将消息发送给符合指定 routing key 的队列,适用于路由模式。 |
Topic | 通配符模式:将消息发送给符合 routing pattern(路由模式)的队列,适用于主题模式。 |
1.4 Routing Key 和 Binding Key
名称 | 描述 |
---|---|
Routing Key | 路由键。生产者在发送消息到交换机时指定的字符串,用于指示交换机如何处理该消息。 |
Binding Key | 绑定键。通过 Binding(绑定)将交换机与队列关联起来,并指定一个绑定键,以便 RabbitMQ 知道如何正确地将消息路由到队列。 |
在大多数情况下,包括官方文档和 RabbitMQ Java API 中,通常会将 Binding Key 和 Routing Key 统称为 Routing Key。为了避免混淆,可以这样理解:
- 在绑定队列时,使用的路由键称为 Binding Key。
- 在发送消息时,使用的路由键称为 Routing Key。
交换机接收到消息后,会将 Routing Key 与所有绑定到该交换机的队列的 Binding Key 进行匹配。如果 Routing Key 与某个 Binding Key 匹配成功,消息将被路由到相应的队列;如果没有匹配的队列,则消息会被丢弃或返回给生产者(具体行为取决于配置)。这种机制确保消息能够被正确分发到符合条件的队列中。
1.5 Routing (路由模式)
路由模式是发布订阅模式的变种,它在发布订阅的基础上增加了 Routing Key 的规则。发布订阅模式会将所有消息无条件地分发给所有消费者,而路由模式则由 Exchange 根据 Routing Key 的规则对消息进行筛选,并将其分发到对应的消费者队列中。该模式适用于需要根据特定规则分发消息的场景。
1.6 Topics (通配符模式)
主题模式(Topics)是路由模式的升级版,在 Routing Key 的基础上增加了通配符匹配功能,使消息分发更加灵活。其基本原理与路由模式相同:生产者将消息发送到交换机,交换机根据 Routing Key 将消息转发到匹配的队列中。不同之处在于 Routing Key 的匹配方式:路由模式使用完全匹配,而主题模式支持通配符匹配,类似正则表达式的方式定义 Routing Key 的模式。该模式适用于需要灵活匹配和过滤消息的场景。
1.7 RPC 通信
在 RPC 通信过程中不再区分生产者和消费者,而是更像传统的 RPC 远程调用。其核心机制是通过两个队列实现消息的请求和响应,从而完成一个带有回调功能的通信过程。
- 客户端将消息发送到指定的请求队列,同时在消息属性中设置 replyTo 字段,该字段用于指定回调队列,回调队列用于接收服务端的响应消息。
- 服务端接收请求后,处理完成并将响应消息发送到 replyTo 字段指定的回调队列。
- 客户端在回调队列中等待响应消息,一旦收到响应,客户端会检查消息的 correlationId 属性,以确保该消息是期望的响应结果。
1.8 Publisher Confirms (发布确认)
Publisher Confirms 模式是 RabbitMQ 提供的一种机制,用于确保消息可靠地发送到 RabbitMQ 服务器。在该模式下,生产者通过调用 channel.confirmSelect() 将通道设置为确认模式后,RabbitMQ 会为发布的每条消息分配一个唯一的序列号。生产者可以将这些序列号与消息关联起来,以便跟踪消息的状态。消息被服务器接收并处理后,RabbitMQ 会异步向生产者发送确认(ACK),包含消息的唯一序列号,表明消息已成功送达。
通过 Publisher Confirms 模式,生产者能够确保消息被 RabbitMQ 服务器成功接收,从而避免消息丢失的问题。这种模式适用于对数据安全性要求较高的场景,例如金融交易和订单处理等领域。
二: 7 种工作模式的简单实现
2.1 简单模式
快速入门程序采用的是简单模式,此处略过详细描述。
2.2 Work Queues (工作队列)
工作队列模式是简单模式的增强版,其区别在于简单模式只有一个消费者,而工作队列模式支持多个消费者同时接收消息。消费者之间存在竞争关系,每条消息只能被其中一个消费者接收和处理。
- 引入依赖
<dependency>
<groupId>com.rabbitmq</groupId>
<artifactId>amqp-client</artifactId>
<version>5.7.3</version>
</dependency>
- 编写生产者代码
工作队列模式与简单模式的区别在于支持多个消费者,因此生产者和消费者的代码差异不大。相比简单模式,生产者的代码基本保持一致。为了直观地展示多个消费者之间的竞争关系,可以一次发送 10 条消息,因此需要将发送消息的逻辑改为批量发送 10 条消息。
public class WorkRabbitProducer {
public static void main(String[] args) throws Exception {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65");
factory.setPort(15673);
factory.setVirtualHost("bite");
factory.setUsername("study");
factory.setPassword("study");
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 声明队列,如果队列不存在,则创建;如果存在,则使用现有队列
channel.queueDeclare("work_queues", true, false, false, null);
// 3. 发送消息
for (int i = 0; i < 10; i++) {
String msg = "Hello World " + i;
channel.basicPublish("", "work_queues", null, msg.getBytes());
System.out.println("消息发送: " + msg);
}
// 4. 释放资源
channel.close();
connection.close();
}
}
- 消费者代码:消费者代码与简单模式一致,只是需要复制两份到两个文件中去启动。两个消费者的代码完全相同。
public class WorkRabbitmqConsumer1 {
public static void main(String[] args) throws Exception {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65"); // IP地址
factory.setPort(15673); // 端口号2
factory.setVirtualHost("bite"); // 虚拟主机名称
factory.setUsername("study"); // 用户名
factory.setPassword("study"); // 密码
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 声明队列,如果没有这样的队列,则自动创建;如果存在,则不创建
channel.queueDeclare("work_queues", true, false, false, null);
// 3. 接收消息并消费
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("work_queues", true, consumer);
}
}
建议先启动两个消费者再启动生产者。如果先启动生产者,由于消息数量较少且处理速度较快,第一个启动的消费者可能会瞬间将10条消息全部消费完。因此,为了更均衡地分配消息,应该先启动两个消费者,再启动生产者。
2.3 Publish / Subscribe (发布 / 订阅)
在发布/订阅模型中引入了一个名为 Exchange 的角色。Exchange 有三种常见类型,每种类型代表不同的路由规则,从而对应不同的工作模式。接下来,我们将了解 Publish / Subscribe 模式的具体实现和特点。
交换机类型 | 描述 |
---|---|
Fanout | 广播模式:将消息发送给所有绑定到交换机的队列,适用于发布/订阅模式。 |
Direct | 定向模式:将消息发送给符合指定 routing key 的队列,适用于路由模式。 |
Topic | 通配符模式:将消息发送给符合 routing pattern(路由模式)的队列,适用于主题模式。 |
- 编写生产者代码,与前面两种模式的区别在于需要创建交换机,并将队列与交换机进行绑定。
public class FanoutRabbitProducer {
public static void main(String[] args) throws Exception {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65"); // IP地址
factory.setPort(15673); // 端口号
factory.setVirtualHost("bite"); // 虚拟主机名称,默认值为 "/"
factory.setUsername("study"); // 用户名
factory.setPassword("study"); // 密码
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 创建交换机
/*
* exchangeDeclare(String exchange, BuiltinExchangeType type, boolean durable,
* boolean autoDelete, boolean internal, Map<String, Object> arguments)
* 参数:
* 1. exchange: 交换机名称
* 2. type: 交换机类型
* * DIRECT("direct"): 定向直连
* * FANOUT("fanout"): 扇形(广播), 每个队列都能收到消息
* * TOPIC("topic"): 通配符模式
* * HEADERS("headers"): 参数匹配(较少使用)
* 3. durable: 是否持久化
* 4. autoDelete: 是否自动删除
* 5. internal: 是否内部使用, 一般为false
* 6. arguments: 参数
*/
channel.exchangeDeclare("fanout_exchange", BuiltinExchangeType.FANOUT, true, false, false, null);
// 3. 声明队列,如果队列不存在则自动创建;如果存在,则不创建
channel.queueDeclare("fanout_queue1", true, false, false, null);
channel.queueDeclare("fanout_queue2", true, false, false, null);
// 4. 绑定队列和交换机
/*
* queueBind(String queue, String exchange, String routingKey, Map<String, Object> arguments)
* 参数:
* 1. queue: 队列名称
* 2. exchange: 交换机名称
* 3. routingKey: 路由key
* 如果交换机类型为fanout,routingKey设置为"",表示每个消费者都能收到所有消息
*/
channel.queueBind("fanout_queue1", "fanout_exchange", "");
channel.queueBind("fanout_queue2", "fanout_exchange", "");
// 5. 发送消息
/*
* basicPublish(String exchange, String routingKey, AMQP.BasicProperties props, byte[] body)
* 参数说明:
* 1. exchange: 交换机名称
* 2. routingKey: 如果交换机类型为fanout,routingKey设置为"",表示每个消费者都能收到所有消息
*/
String msg = "hello fanout";
channel.basicPublish("fanout_exchange", "", null, msg.getBytes());
System.out.println("消息发送: " + msg);
// 6. 释放资源
channel.close();
connection.close();
}
}
- 编写消费者代码,消费者 1 代码如下,消费者 2 的代码修改一下队列名称即可,此处省略
public class FanoutRabbitmqConsumer1 {
public static void main(String[] args) throws Exception {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65"); // IP地址
factory.setPort(15673); // 端口号
factory.setVirtualHost("bite"); // 虚拟主机名称
factory.setUsername("study"); // 用户名
factory.setPassword("study"); // 密码
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 接收消息并消费
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("fanout_queue1", true, consumer);
}
}
2.4 Routing (路由模式)
在队列和交换机的绑定中,必须指定一个 BindingKey。消息发送方在向 Exchange 发送消息时,需要指定消息的 RoutingKey。Exchange 不会将消息交给所有绑定的队列,而是根据消息的 RoutingKey 进行匹配,只有当队列绑定的 BindingKey 与消息的 RoutingKey 完全一致时,队列才能接收到消息。
- 编写生产者代码
public class DirectRabbitProducer {
public static void main(String[] args) throws Exception {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65"); // IP地址
factory.setPort(15673); // 端口号
factory.setVirtualHost("bite"); // 虚拟主机名称,默认值为 "/"
factory.setUsername("study"); // 用户名
factory.setPassword("study"); // 密码
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 创建交换机
channel.exchangeDeclare("direct_exchange", BuiltinExchangeType.DIRECT, true, false, false, null);
// 3. 声明队列
channel.queueDeclare("direct_queue1", true, false, false, null);
channel.queueDeclare("direct_queue2", true, false, false, null);
// 4. 绑定队列和交换机
channel.queueBind("direct_queue1", "direct_exchange", "orange"); // 队列1绑定到 RoutingKey 为 "orange"
channel.queueBind("direct_queue2", "direct_exchange", "black"); // 队列2绑定到 RoutingKey 为 "black"
channel.queueBind("direct_queue2", "direct_exchange", "green"); // 队列2绑定到 RoutingKey 为 "green"
// 5. 发送消息
String msgOrange = "hello direct, I am orange";
channel.basicPublish("direct_exchange", "orange", null, msgOrange.getBytes());
System.out.println("发送消息:" + msgOrange);
String msgBlack = "hello direct, I am black";
channel.basicPublish("direct_exchange", "black", null, msgBlack.getBytes());
System.out.println("发送消息:" + msgBlack);
String msgGreen = "hello direct, I am green";
channel.basicPublish("direct_exchange", "green", null, msgGreen.getBytes());
System.out.println("发送消息:" + msgGreen);
// 6. 释放资源
channel.close();
connection.close();
}
}
- 编写消费者代码
在 Routing 模式下,消费者代码与 Publish/Subscribe 模式的消费者代码相同,可以直接复制两份,分别命名为 DirectRabbitmqConsumer1 和 DirectRabbitmqConsumer2。只需修改对应的队列名称即可实现不同的消费逻辑。
public class DirectRabbitmqConsumer1 {
public static void main(String[] args) throws Exception {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65"); // IP地址
factory.setPort(15673); // 端口号
factory.setVirtualHost("bite"); // 虚拟主机名称,默认值为 "/"
factory.setUsername("study"); // 用户名
factory.setPassword("study"); // 密码
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 接收消息并消费
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));
}
};
// 监听队列 direct_queue1
channel.basicConsume("direct_queue1", true, consumer);
}
}
2.5 Topics (通配符模式)
Topics 模式与 Routing 模式的主要区别在于交换机类型和匹配规则:Topics 模式使用的交换机类型为 topic,而 Routing 模式使用的交换机类型为 direct。此外,topic 类型的交换机在匹配规则上进行了扩展,支持使用通配符进行匹配,而 direct 类型的交换机要求 BindingKey 和 RoutingKey 完全匹配。
在 topic 类型的交换机中,匹配规则有以下要求:RoutingKey 是由点 (.) 分隔的一系列单词,例如 “stock.usd.nyse”、“nyse.vmw”。BindingKey 与 RoutingKey 格式相同,也是由点分隔的字符串。同时,BindingKey 支持两种特殊字符进行模糊匹配:* 表示一个单词,# 表示多个单词(0 到 N 个)。
Binding Key | 路由队列 | 说明 |
---|---|---|
d.a.b | Q1 和 Q2 | 同时匹配到 Q1 和 Q2 |
d.a.f | Q1 | 仅匹配到 Q1 |
c.e.f | Q2 | 仅匹配到 Q2 |
d.b.f | 丢弃或返回生产者 | 不匹配任何队列,需设置 mandatory 参数才能返回给生产者,否则直接丢弃 |
- 生产者代码
public class TopicRabbitProducer {
public static void main(String[] args) throws Exception {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65"); // IP地址,默认值为localhost
factory.setPort(15673); // 端口号,默认值为5672
factory.setVirtualHost("bite"); // 虚拟主机名称,默认值为 "/"
factory.setUsername("study"); // 用户名,默认值为guest
factory.setPassword("study"); // 密码,默认值为guest
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 创建交换机
channel.exchangeDeclare("topic_exchange", BuiltinExchangeType.TOPIC, true, false, false, null);
// 3. 声明队列
channel.queueDeclare("topic_queue1", true, false, false, null);
channel.queueDeclare("topic_queue2", true, false, false, null);
// 4. 绑定队列和交换机
// 队列1绑定到 "*.error",仅接收包含 "error" 的消息
channel.queueBind("topic_queue1", "topic_exchange", "*.error");
// 队列2绑定到 "#.info" 和 "*.error",接收包含 "info" 或 "error" 的消息
channel.queueBind("topic_queue2", "topic_exchange", "#.info");
channel.queueBind("topic_queue2", "topic_exchange", "*.error");
// 5. 发送消息
String msgError = "hello topic, I'm order.error";
channel.basicPublish("topic_exchange", "order.error", null, msgError.getBytes());
System.out.println("发送消息:" + msgError);
String msgInfo = "hello topic, I'm order.pay.info";
channel.basicPublish("topic_exchange", "order.pay.info", null, msgInfo.getBytes());
System.out.println("发送消息:" + msgInfo);
String msgPayError = "hello topic, I'm pay.error";
channel.basicPublish("topic_exchange", "pay.error", null, msgPayError.getBytes());
System.out.println("发送消息:" + msgPayError);
// 6. 释放资源
channel.close();
connection.close();
}
}
- 编写消费者代码
在 Topic 模式中,消费者代码与 Routing 模式的消费者代码相同,只需修改消费的队列名称即可。可以分别创建两个消费者,命名为 TopicRabbitmqConsumer1 和 TopicRabbitmqConsumer2,并将它们的队列名称分别修改为对应的队列。
public class TopicRabbitmqConsumer1 {
public static void main(String[] args) throws Exception {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65"); // IP地址,默认值为localhost
factory.setPort(15673); // 端口号,默认值为5672
factory.setVirtualHost("bite"); // 虚拟主机名称,默认值为 "/"
factory.setUsername("study"); // 用户名,默认值为guest
factory.setPassword("study"); // 密码,默认值为guest
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 接收消息并消费
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));
}
};
// 监听队列 topic_queue1
channel.basicConsume("topic_queue1", true, consumer);
}
}
2.6 RPC 通信
RPC 是一种通过网络请求远程计算机上的服务,无需了解底层网络实现的技术,类似于 HTTP 的远程调用。使用 RabbitMQ 实现 RPC 通信时,通常通过两个队列完成一个可回调的过程。
- 客户端将消息发送到指定的请求队列,同时在消息属性中设置 replyTo 字段,该字段用于指定回调队列,回调队列用于接收服务端的响应消息。
- 服务端接收请求后,处理完成并将响应消息发送到 replyTo 字段指定的回调队列。
- 客户端在回调队列中等待响应消息,一旦收到响应,客户端会检查消息的 correlationId 属性,以确保该消息是期望的响应结果。
- 编写客户端代码,客户端:
步骤 | 流程描述 |
---|---|
1 | 声明两个队列,其中一个为回调队列 replyQueueName,并声明当前请求的唯一标识 corrId。 |
2 | 将 replyQueueName 和 corrId 配置到要发送的消息队列中,供服务端处理并回传结果。 |
3 | 使用阻塞队列阻塞当前进程,监听回调队列中的消息,并将请求放入阻塞队列中等待响应。 |
4 | 当阻塞队列收到消息时,主线程被唤醒,获取响应内容并打印返回结果。 |
public class RPCClient {
public static void main(String[] args) throws Exception {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65"); // IP地址,默认值为localhost
factory.setPort(15673); // 端口号,默认值为5672
factory.setVirtualHost("bite"); // 虚拟主机名称,默认值为 "/"
factory.setUsername("study"); // 用户名,默认值为guest
factory.setPassword("study"); // 密码,默认值为guest
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 声明请求队列
channel.queueDeclare("rpc_request_queue", true, false, false, null);
// 3. 唯一标识本次请求
String corrId = UUID.randomUUID().toString(); // 生成唯一标识符
// 4. 声明一个临时队列用于回调,并获取其名称
String replyQueueName = channel.queueDeclare().getQueue();
// 5. 生成发送消息的属性
/*
* BasicProperties:用于设置消息的附加属性
* - correlationId: 唯一标识本次请求
* - replyTo: 回调队列的名称
*/
AMQP.BasicProperties props = new AMQP.BasicProperties.Builder()
.correlationId(corrId) // 设置唯一标识符
.replyTo(replyQueueName) // 设置回调队列名称
.build();
// 6. 发送消息到指定队列
String message = "hello rpc...";
channel.basicPublish("", "rpc_request_queue", props, message.getBytes());
System.out.println(" [RPCClient] Sent: " + message);
// 7. 使用阻塞队列存储回调结果
final BlockingQueue<String> response = new ArrayBlockingQueue<>(1);
// 8. 接收服务端的响应
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));
// 判断响应的 correlationId 是否与请求的 corrId 一致
if (properties.getCorrelationId().equals(corrId)) {
response.offer(new String(body, "UTF-8")); // 将响应消息放入阻塞队列
}
}
};
// 监听回调队列
channel.basicConsume(replyQueueName, true, consumer);
// 9. 从阻塞队列中获取响应结果
String result = response.take(); // 阻塞直到接收到响应消息
System.out.println(" [RPCClient] Result: " + result);
// 10. 释放资源
channel.close();
connection.close();
}
}
- 编写服务器代码:
import com.rabbitmq.client.*;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class RPCServer {
public static void main(String[] args) throws IOException, TimeoutException {
// 1. 创建连接工厂并设置参数
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("110.41.51.65"); // IP地址,默认值为localhost
factory.setPort(15673); // 端口号,默认值为5672
factory.setVirtualHost("bite"); // 虚拟主机名称,默认值为 "/"
factory.setUsername("study"); // 用户名,默认值为guest
factory.setPassword("study"); // 密码,默认值为guest
// 创建连接和通道
Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
// 2. 声明请求队列
channel.queueDeclare("rpc_request_queue", true, false, false, null);
// 设置同时最多只能处理一个消息
channel.basicQos(1);
System.out.println(" [RPCServer] Awaiting RPC requests");
// 3. 创建消费者并监听请求队列
Consumer consumer = new DefaultConsumer(channel) {
@Override
public void handleDelivery(String consumerTag, Envelope envelope,
AMQP.BasicProperties properties, byte[] body) throws IOException {
// 生成返回消息的属性
/*
* AMQP.BasicProperties:消息的属性
* - correlationId: 对应请求的唯一标识符
*/
AMQP.BasicProperties replyProps = new AMQP.BasicProperties
.Builder()
.correlationId(properties.getCorrelationId()) // 关联请求的唯一标识
.build();
// 获取请求内容
String message = new String(body);
System.out.println(" [RPCServer] Received request: " + message);
// 生成响应内容
String response = "request: " + message + ", response: 处理成功";
// 回复消息到请求方的回调队列
channel.basicPublish("", properties.getReplyTo(), replyProps, response.getBytes());
System.out.println(" [RPCServer] Sent response: " + response);
// 对消息进行手动应答
/*
* basicAck(long deliveryTag, boolean multiple)
* 参数:
* - deliveryTag: 当前消息的标识
* - multiple: 是否批量应答
*/
channel.basicAck(envelope.getDeliveryTag(), false);
}
};
// 消费请求队列中的消息
/*
* basicConsume(String queue, boolean autoAck, Consumer callback)
* 参数:
* - queue: 队列名称
* - autoAck: 是否自动确认
* - callback: 消费者逻辑
*/
channel.basicConsume("rpc_request_queue", false, consumer);
}
}
2.7 Publisher Confirms (发布确认)
消息中间件都会会面临消息丢失的问题,主要分为以下三种情况:
问题类型 | 描述 | 解决方案 |
---|---|---|
生产者问题 | 由于应用程序故障、网络抖动等原因,生产者未能成功将消息发送到消息中间件(Broker)。 | 采用发布确认(Publisher Confirms)机制实现。 |
Broker问题 | 生产者成功将消息发送到 Broker,但由于 Broker 自身未能妥善保存消息,导致消息丢失。 | 使用消息持久化机制。 |
消费者问题 | Broker 将消息发送到消费者后,消费者在处理消息时出错,导致 Broker 将未正确消费的消息从队列中删除,从而丢失消息。 | 使用消息应答机制。 |
发布确认是 RabbitMQ 的七大工作模式之一。生产者将信道设置为确认模式后,该信道上发布的每条消息都会被分配一个唯一的 ID(从 1 开始)。当消息成功投递到所有匹配的队列后,RabbitMQ 会发送一个包含消息唯一 ID 的确认消息给生产者,告知消息已正确到达目标队列。如果消息和队列是持久化的,确认消息会在将消息写入磁盘后发送。确认消息的 deliveryTag 表示消息的序号,同时,Broker 可以通过设置 channel.basicAck 方法的 multiple 参数,表示该序号之前的所有消息都已成功处理。
发送方确认机制的最大优势在于其异步性,生产者可以同时发布消息并等待信道返回确认消息。当消息被确认后,生产者可以通过回调方法处理确认消息;如果 RabbitMQ 因内部错误导致消息丢失,则会发送一条 nack(Basic.Nack)命令,生产者同样可以在回调方法中处理该 nack 命令。使用发送确认机制必须将信道设置为确认模式,这是 AMQP 0.9.1 协议的扩展功能,默认情况下未启用,生产者需通过 channel.confirmSelect() 将信道设置为确认模式,发布确认有 3 种策略,接下来我们来学习这三种策略。
2.7.1 Publishing Messages Individually (单独确认)
static void publishMessagesIndividually() throws Exception {
try (Connection connection = createConnection()) {
// 创建信道
Channel channel = connection.createChannel();
// 开启信道确认模式
channel.confirmSelect();
// 声明队列
channel.queueDeclare(PUBLISHER_CONFIRMS_QUEUE_NAME1, true, false, true, null);
long start = System.currentTimeMillis();
// 循环发送消息
for (int i = 0; i < 200; i++) {
String body = "消息 " + i;
// 发布消息
channel.basicPublish("", PUBLISHER_CONFIRMS_QUEUE_NAME1, null, body.getBytes());
// 等待确认消息
channel.waitForConfirmsOrDie(5_000);
}
long end = System.currentTimeMillis();
System.out.format("Published %d messages individually in %d ms%n", MESSAGE_COUNT, end - start);
}
}
可以发现发送 200 条消息耗时较长,这是因为上述代码在每发送一条消息后都会调用 channel.waitForConfirmsOrDie 方法,等待服务端的确认。这种方式实际上是一种串行同步等待的策略,尤其对于持久化消息,还需要等待消息被存储到磁盘后才能返回。为了解决这个问题,RabbitMQ 提供了发布确认机制的异步操作,可以在发送消息的同时异步等待消息确认,从而大幅提升效率。
2.7.2 Publishing Messages in Batches (批量确认)
每发送一批消息后,调用 channel.waitForConfirms 方法等待服务器返回确认结果。
static void publishMessagesInBatch() throws Exception {
// 创建连接
try (Connection connection = createConnection()) {
// 创建信道
Channel channel = connection.createChannel();
channel.confirmSelect();
// 声明队列
channel.queueDeclare(PUBLISHER_CONFIRMS_QUEUE_NAME2, true, false, true, null);
// 设置批量大小
int batchSize = 100; // 每次确认的消息数量
int outstandingMessageCount = 0; // 记录未确认的消息数量
long start = System.currentTimeMillis(); // 记录开始时间
// 循环发送消息
for (int i = 0; i < MESSAGE_COUNT; i++) {
// 生成消息内容
String body = "消息 " + i;
// 发布消息
channel.basicPublish("", PUBLISHER_CONFIRMS_QUEUE_NAME2, null, body.getBytes());
outstandingMessageCount++; // 增加未确认消息数量
// 批量确认消息
if (outstandingMessageCount == batchSize) {
channel.waitForConfirmsOrDie(5_000); // 等待确认,超时时间为 5000 毫秒
outstandingMessageCount = 0; // 清空未确认消息数量
}
}
// 处理剩余未确认的消息
if (outstandingMessageCount > 0) {
channel.waitForConfirmsOrDie(5_000); // 等待确认
}
long end = System.currentTimeMillis(); // 记录结束时间
// 打印消息发送耗时
System.out.format("Published %d messages in batch in %d ms%n", MESSAGE_COUNT, end - start);
}
}
相比于单独确认策略,批量确认显著提升了确认的效率,但其缺点在于当出现 Basic.Nack 或超时时,无法明确是哪条消息出现问题,客户端需要重新发送整批消息,这可能导致明显的重复消息数量。如果消息频繁丢失,批量确认的性能可能不增反降。
2.7.3 Handling Publisher Confirms Asynchronously (异步确认)
异步确认的编程实现相对复杂。Channel 接口提供了 addConfirmListener 方法,用于添加 ConfirmListener 回调接口。ConfirmListener 接口包含两个方法:handleAck(long deliveryTag, boolean multiple) 和 handleNack(long deliveryTag, boolean multiple),分别用于处理 RabbitMQ 发送给生产者的 ack 和 nack 消息。其中,deliveryTag 表示消息的序号,multiple 表示是否批量确认。
为每个信道维护一个已发送消息的序号集合是必要的。当收到 RabbitMQ 的确认回调时,从集合中删除对应的消息。在开启确认模式后,信道上发送的每条消息都会附带一个从 1 开始递增的 deliveryTag 序号。可以利用 SortedSet 的有序性来维护这些已发送的消息序号,从而高效管理确认状态。
情况 | 处理逻辑 |
---|---|
收到 ack | 从序列中删除对应消息的序号。如果是批量确认,则表示小于等于当前序号 deliveryTag 的所有消息都已确认,需清除对应的集合。 |
收到 nack | 处理逻辑与 ack 类似,但需要结合具体业务场景,对未确认的消息进行处理,如消息重发等操作。 |
static void handlePublishConfirmsAsynchronously() throws Exception {
try (Connection connection = createConnection()) {
// 创建信道
Channel channel = connection.createChannel();
// 声明队列
channel.queueDeclare(PUBLISHER_CONFIRMS_QUEUE_NAME3, false, false, true, null);
// 开启信道确认模式
channel.confirmSelect();
// 创建有序集合,用于存储未确认的消息序号,确保消息的顺序性
SortedSet<Long> confirmSet = Collections.synchronizedSortedSet(new TreeSet<>());
// 添加确认监听器
channel.addConfirmListener(new ConfirmListener() {
// 处理 ack 消息
@Override
public void handleAck(long deliveryTag, boolean multiple) throws IOException {
if (multiple) {
// 批量确认:清除集合中小于等于 deliveryTag 的消息序号
confirmSet.headSet(deliveryTag + 1).clear();
} else {
// 单条确认:移除当前消息序号
confirmSet.remove(deliveryTag);
}
}
// 处理 nack 消息
@Override
public void handleNack(long deliveryTag, boolean multiple) throws IOException {
System.err.format("deliveryTag: %d, multiple: %b%n", deliveryTag, multiple);
if (multiple) {
// 批量确认失败:清除集合中小于等于 deliveryTag 的消息序号
confirmSet.headSet(deliveryTag + 1).clear();
} else {
// 单条确认失败:移除当前消息序号
confirmSet.remove(deliveryTag);
}
// 如果处理失败,这里需要添加消息重发逻辑(具体实现可根据业务需求补充)
}
});
// 循环发送消息
long start = System.currentTimeMillis();
for (int i = 0; i < MESSAGE_COUNT; i++) {
String message = "消息" + i;
// 获取下次发送消息的序号(从 1 开始递增)
long nextPublishSeqNo = channel.getNextPublishSeqNo();
// 发布消息
channel.basicPublish("", PUBLISHER_CONFIRMS_QUEUE_NAME3, null, message.getBytes());
// 将消息序号存入集合
confirmSet.add(nextPublishSeqNo);
}
// 等待所有消息确认完毕
while (!confirmSet.isEmpty()) {
Thread.sleep(10); // 每隔 10 毫秒检查一次
}
long end = System.currentTimeMillis();
System.out.format("Published %d messages and handled confirms asynchronously in %d ms%n", MESSAGE_COUNT, end - start);
}
}