RabbitMQ笔记-4

这里记录官方的第3个Demo,这个Demo主要讲了交换器在MQ中的使用

[TOC]

To-do List

  • 文章概要
  • 交换器的类型
    • 不使用交换器
    • fanout交换器
    • direct交换器
    • topic交换器
    • headers交换器

0X00 概要,以及记录完成情况

  • 在前两个Demo中生产者消费者是通过默认的交换器,也就是空字符串(channel.basicPublish()中的exchange参数为””),前面有解释过当exchange参数为空字符串的情况下,消息是通过routingKey指定的名称路由到队列(如果存在,是直接路由到队列),并没有通过交换器,而MQ中消息传递模型核心思想是生产者永远不会降任何消息直接发送到队列.实际上,生产者通常甚至不知道消息是否会被传递到任何队列。

  • 相反,生产者只能向交换器发送消息,而交换器处理起来也非常的简单.一方面它接收来自生产者的消息,另一方面它将接收到的这些消息推送到队列,交换器呢也需要知道它应该如何去处理它收到的消息它应该附加到特定队列吗?它应该附加到许多队列吗?或者它应该被丢弃,交换器的规则是由交换器的类型去定义

0X01 交换器的类型

默认交换器,或者叫不使用交换器

  • 生产者
1
2
//消息通过routingKey指定的名称路由到队列(如果存在,是直接路由到队列)
channel.basicPublish("", QUEUE_NAME, null, message.getBytes(StandardCharsets.UTF_8));

  • 消费者
1
channel.queueDeclare(QUEUE_NAME, false, false, false, null);

  • 总结: 在默认交换器的情况下,生产者是直接将消息推送在队列,但是官方不推荐这样写

fanout类型的交换器

  • 下图展示了fanout交换器的示意图

fanout示意图


  • fanout交换器也叫扇形交换器,它会将收到的消息广播给所有的和它绑定的队列,在上图中P只关心消息发送到哪个交换器,由交换器X去觉得把消息放到哪个队列,二C3 C4只关心自己订阅了哪个队列

生产者代码编写
  • 下面写一个完整的fanout交换器例子- 生产者
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
package demo03;

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.TimeoutException;

/**
* 官方的第三个Demo 生产者
* 主要介绍了以下功能:
* 1.Fanout交换器的使用
*/
public class FanoutSend {

/**
* 定义交换器名称
*/
private static final String EXCHANGE_NAME = "logs";

public static void main(String[] args) {

//配置连接信息
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
factory.setUsername("admin");
factory.setPassword("123456");

//try-with-resources
try (Connection connection = factory.newConnection();
Channel channel = connection.createChannel();
) {

//声明交换器类型
channel.exchangeDeclare(EXCHANGE_NAME, "fanout");

//定义消息
String msg = "你好";

//通过通道把消息发送给交换器
channel.basicPublish(EXCHANGE_NAME, "", null, msg.getBytes(StandardCharsets.UTF_8));

System.out.println(" [X] Sent '" + msg + "'");

} catch (TimeoutException | IOException e) {
e.printStackTrace();
}
}

}

消费者代码编写
  • 下面写一个完整的fanout交换器例子- 消费者
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
package demo03;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.TimeoutException;

/**
* 对应官方Demo03的客户端
*/
public class FanoutRecv {

/**
* 交换器名称
*/
private static final String EXCHANGE_NAME = "logs";

public static void main(String[] args) throws IOException, TimeoutException {

//连接信息
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
factory.setUsername("admin");
factory.setPassword("123456");

Connection connection = factory.newConnection();
Channel channel = connection.createChannel();

//通道声明交换器
channel.exchangeDeclare(EXCHANGE_NAME, "fanout");

//通道声明一个不为持久的队,独占的(仅限此连接),自动删除的(服务器不在使用或者下线将此队列删除)队列
String queueName = channel.queueDeclare().getQueue();

//将队列绑定到交换器
channel.queueBind(queueName, EXCHANGE_NAME, "");

System.out.println(" [*] 正在等待消息,退出请按 CTRL+C");

//消息确认,关闭自动确认

// DeliverCallback xx = new DeliverCallback() {
// @Override
// public void handle(String s, Delivery delivery) throws IOException {
// String msg = new String(delivery.getBody(), StandardCharsets.UTF_8);
// System.out.println(" [X] 收到消息'" + msg + "',我的工作编号为 FanoutRecv");
// channel.basicAck(delivery.getEnvelope().getDeliveryTag(), false);
// }
// };
// channel.basicConsume(queueName, false, xx, consumerTag -> {});

DeliverCallback deliverCallback = (consumerTag, delivery) -> {
String msg = new String(delivery.getBody(), StandardCharsets.UTF_8);
System.out.println(" [X] 收到消息'" + msg + ",我的工作编号为 FanoutRecv'");
channel.basicAck(delivery.getEnvelope().getDeliveryTag(), false);
};
channel.basicConsume(queueName, false, deliverCallback, consumerTag -> {});
}

}

代码中的一些API介绍(基于5.6.0 AP)

生产者的代码

channel.exchangeDeclare()

channel.exchangeDeclare(java.lang.String exchange, java.lang.String type)

主动声明一个非自动删除,非持久的交换,没有额外的参数

参数名 参数类型 参数意义
exchange String 交换器的名称(比如上面生产者代码中的”logs”)
type String 定义交换器的类型(代码中的”fanout”类型交换器)

channel.basicPublish()

channel.basicPublishjava.lang.String exchange, java.lang.String routingKey, AMQP.BasicProperties props, byte[] body)

参数名 参数类型 参数意义
exchange String 消息需要发送到的交换器名称(比如上面生产者代码中的”logs”)
routingKek String 路由器密钥
props AMQP.BasicProperties 消息的其他属性 - 路由头等
body byte[] 消息正文

消费者的代码

channel.basicAck()

basicAck(long deliveryTag, boolean multiple)

参数名 参数类型 参数意义
deliveryTag long RabbitMQ推送消息给消费者时,会附带一个DeliveryTag,以便消费者可以在消息确认时告诉 RabbitMQ 到底是哪条消息被确认了。RabbitMQ 保证在每个信道中,每条消息的 Delivery Tag 从 1 开始递增
multiple boolean 取值为 false 时,表示通知 RabbitMQ 当前消息被确认;如果为 true,则额外将比第一个参数指定的 delivery tag 小的消息一并确认。(批量确认针对的是整个信道,参考gordon.study.rabbitmq.ack.TestBatchAckInOneChannel.java。) 对同一消息的重复确认,或者对不存在的消息的确认,会产生 IO 异常,导致信道关闭。

channel.basicConsume()

basicConsume(java.lang.String queue, boolean autoAck, DeliverCallback deliverCallback, CancelCallback cancelCallback)

参数名 参数类型 参数意义
queue String 队列名称
autoAck boolean 是否自动确认消息(手动确认和自动确认,在开发中一般为手动确认,即值为false,详细说明参考Consumer Acknowledgements and Publisher Confirms)
deliverCallback deliverCallback 传递消息时的回调
cancelCallback CancelCallback 取消消费者时的回调(这个是个功能性接口可以用作lambda表达式或方法引用的赋值目标)

direct类型的交换器

  • 下图展示了direct交换器的原理图

direct交换器原理图


  • 上图可以看到 direct类型的交换器x和两个通道绑定,分别是Q1和Q2,第一个队列Q1绑定orange路由key(routingKey),而Q2则绑定的blackgreen路由key

  • 在这种配置下P(生产者)发送消息到X(direct交换器),交换器根据消息的routingKey(路由key)的不同将消息分配到不同的队列(Q1,Q2)

  • 图中orangekey的消息被分配到了Q1,因为Q1队列绑定的orange;而black,green被交换器分配到了Q2,因为Q2队列绑定了blackgreen路由key

  • 使用相同的路由key去绑定多个队列是完全没有问题的,如下图所示

相同路由key绑定多个队列

  • 上图中Q1,Q2都绑定了路由keyblack,在这种情况下,direct交换器类型就表现的和fanout类型一样了
生产者代码编写
  • 下面为Driect类型的交换器的生产者代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
package demo04;

import com.rabbitmq.client.BuiltinExchangeType;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.TimeoutException;

/**
* 官方的第4个Demo
* 主要实现了以下内容:
* 1.direct类型交换器的使用
*/
public class DirectSend {

/**
* 定义交换器的名称
*/
public static final String EXCHANGE_NAME = "logs";

public static void main(String[] args) {

//配置连接信息
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
factory.setUsername("admin");
factory.setPassword("123456");

//try-with-resources
try (Connection connection = factory.newConnection();
Channel channel = connection.createChannel()
) {

//声明交换器类型
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT);

//定义Info消息(消费者只有在routingKek相同时才能接收到消息)
String infoMsg = "你好 direct";

//定义Error消息
String errorMsg = "错误 direct";

//消息发送
channel.basicPublish(EXCHANGE_NAME, "info", null, infoMsg.getBytes(StandardCharsets.UTF_8));

System.out.println(" [X] Sent '" + infoMsg + "'");

//错误消息发送
channel.basicPublish(EXCHANGE_NAME, "error", null, errorMsg.getBytes(StandardCharsets.UTF_8));

System.out.println(" [X] Sent '" + errorMsg + "'");

} catch (TimeoutException | IOException e) {
e.printStackTrace();
}

}

}
消费者代码编写
负责接收Info路由key
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
package demo04;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.TimeoutException;

/**
* 对应官方Demo04的客户端
* 负责接收Info
*/
public class DirectRecvForInfo {

/**
* 交换器名称
*/
public static final String EXCHANGE_NAME = "logs";

public static void main(String[] args) throws IOException, TimeoutException {

//配置连接信息
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
factory.setUsername("admin");
factory.setPassword("123456");

Connection connection = factory.newConnection();
Channel channel = connection.createChannel();

//声明交换器
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT);

//通过通道声明一个不为持久的队,独占的(仅限此连接),自动删除的(服务器不在使用或者下线将此队列删除)队列
String queueName = channel.queueDeclare().getQueue();

//绑定到交换器
channel.queueBind(queueName, EXCHANGE_NAME, "info");

System.out.println(" [*] 正在等待消息,退出请按 CTRL+C");

DeliverCallback deliverCallback = (consumerTar, delivery) -> {
String msg = new String(delivery.getBody(), StandardCharsets.UTF_8);
System.out.println(" [X] 收到消息'" + msg + ",我的工作编号为 DirectRecv,负责接收 Info 消息'");
channel.basicAck(delivery.getEnvelope().getDeliveryTag(), false);
};
channel.basicConsume(queueName, false, deliverCallback, consumerTag -> {});
}
}

负责接收Error路由key
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
package demo04;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.TimeoutException;

/**
* 对应官方Demo04的客户端
* 负责接error
*/
public class DirectRecvForError {

/**
* 交换器名称
*/
public static final String EXCHANGE_NAME = "logs";

public static void main(String[] args) throws IOException, TimeoutException {

//配置连接信息
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
factory.setUsername("admin");
factory.setPassword("123456");

Connection connection = factory.newConnection();
Channel channel = connection.createChannel();

//声明交换器
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.DIRECT);

//通过通道声明一个不为持久的队,独占的(仅限此连接),自动删除的(服务器不在使用或者下线将此队列删除)队列
String queueName = channel.queueDeclare().getQueue();

//绑定到交换器
channel.queueBind(queueName, EXCHANGE_NAME, "error");

System.out.println(" [*] 正在等待消息,退出请按 CTRL+C");

DeliverCallback deliverCallback = (consumerTar, delivery) -> {
String msg = new String(delivery.getBody(), StandardCharsets.UTF_8);
System.out.println(" [X] 收到消息'" + msg + ",我的工作编号为 DirectRecv,负责接收 Error 消息'");
channel.basicAck(delivery.getEnvelope().getDeliveryTag(), false);
};
channel.basicConsume(queueName, false, deliverCallback, consumerTag -> {});
}
}
代码效果展示
  • 生产者发送消息

Direct类型交换器生产者发送消息

  • 消费者负责Info类型接收消息

Direct类型交换器Info消费者接受消息

  • 消费者负责Error类型接收消息

Direct类型交换器消费者接受Error消息

  • 这里说一下上面介绍Direct交换器没有说到的地方就拿这个代码举例,当直接启动生产者发送消息,不启动两个消费者消息会丢失,只启动一个消费者,发送给另一个消费者的消息会丢失
代码中的一些API介绍(基于5.6.0 AP)
  • 这里就暂时只说一下上面没说的,有空我会把API整合到一个地方
消费者的代码

channel.queueBind()

channel.queueBind(java.lang.String queue, java.lang.String exchange, java.lang.String routingKey)

主动声明一个非自动删除,非持久的交换,没有额外的参数

参数名 参数类型 参数意义
queue String 通道名称
exchange String 交换器名称
routingKey String 路由Key

topic类型的交换器

  • 下图展示了topic交换器的原理

topic原理图

  • 消息分发规则: 一个附带特殊的选择键将会被转发到绑定键与之匹配的队列中。

  • 路由key规则:路由key必须是由点隔开的一系列的标识符组成:”stock.usd.nyse”,“nyse.vmw”,”quick.orange.rabbit”.你可以定义任数量的标识符,上限为255个字节。

  • 通道绑定的key: 这个有点类似于正则表达式的意思了,如下表:
表达式 匹配意义
* (星号)可以替代一个单词(或者标识符)
# (hash)可以替换零个或多个单词(或者标识符)

  • 图示意义解释: 消息1(fast.orange.)通过交换器(topic类型)分配到消费者Q6(#)和Q7(.orange.*),因为Q6的#(hash)可以匹配任意标识符,而Q7的orange.(星号)匹配前后的一个单词或者标识符,所以这两个通道都能收到消息,消息2(lazy.orange.a.b)通过交换器会到达Q6.Q8,因为Q6匹配所有,而Q8(lazy.#)可以匹配lazy.后面的零个或者多个标识符
生产者代码编写
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
package demo05;

import com.rabbitmq.client.BuiltinExchangeType;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.TimeoutException;


public class TopicSend {

//定义交换器名称
private static final String EXCHANGE_NAME = "topic_logs";

public static void main(String[] args) {

//配置连接信息
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
factory.setUsername("admin");
factory.setPassword("123456");

//tyr-witch-resources
try (Connection connection = factory.newConnection(); Channel channel = connection.createChannel();)
{

//声明交换器类型
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.TOPIC);

//定义消息
String msg = "# 匹配";

//消息发送
channel.basicPublish(EXCHANGE_NAME, "anonymous.info", null, msg.getBytes(StandardCharsets.UTF_8));

System.out.println(" [X] Sent '" + msg + "'");

} catch (TimeoutException | IOException e) {
e.printStackTrace();
}
}

}
消费者代码编写
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
package demo05;

import com.rabbitmq.client.*;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.TimeoutException;

public class TopicRecvForJinHao {

/**
* 交换器名称
*/
public static final String EXCHANGE_NAME = "topic_logs";

public static void main(String[] args) throws IOException, TimeoutException {

//配置连接信息
ConnectionFactory factory = new ConnectionFactory();
factory.setHost("127.0.0.1");
factory.setUsername("admin");
factory.setPassword("123456");

Connection connection = factory.newConnection();
Channel channel = connection.createChannel();

//声明交换器
channel.exchangeDeclare(EXCHANGE_NAME, BuiltinExchangeType.TOPIC);

//通过通道声明一个不为持久的队,独占的(仅限此连接),自动删除的(服务器不在使用或者下线将此队列删除)队列
String queueName = channel.queueDeclare().getQueue();

//绑定到交换器,配置匹配key为 # 接收所有类型消息
channel.queueBind(queueName, EXCHANGE_NAME, "#");

System.out.println(" [*] 正在等待消息,退出请按 CTRL+C");

DeliverCallback deliverCallback = (consumerTar, delivery) -> {
String msg = new String(delivery.getBody(), StandardCharsets.UTF_8);
System.out.println(" [X] 收到消息'" + msg + ",我的路由Key为 # ,负责接收所有消息'");
channel.basicAck(delivery.getEnvelope().getDeliveryTag(), false);
};
channel.basicConsume(queueName, false, deliverCallback, consumerTag -> {});
}

}
代码效果展示
  • 生产者发送消息如图所示
    topic发送消息

  • 消费者#号接收消息
    topic消费者#号匹配消息

其它说明
  • 上面的代码只是一个最简单的demo状态而且只是实验了 # 匹配而已,你们可以启动两个生产者并且使用不同的路由key,然后在启动两个消费者使用不同的匹配规则,就可以实现类似于示例图的效果了
0%