Hi~!这里是奋斗的明志,很荣幸您能阅读我的文章,诚请评论指点,欢迎欢迎 ~~
🌱🌱个人主页:奋斗的明志
🌱🌱所属专栏:RabbitMQ
📚本系列文章为个人学习笔记,在这里撰写成文一为巩固知识,二为展示我的学习过程及理解。文笔、排版拙劣,望见谅。
data:image/s3,"s3://crabby-images/44af2/44af2046549788e3071c8a76fe0ebf51fd12b838" alt=""
发布订阅模式、路由模式、通配符模式
一、Publish/Subscribe(发布/订阅)
在发布/订阅模型中,多了一个
Exchange
角色.
Exchange
常见有三种类型, 分别代表不同的路由规则a)
Fanout
:广播,将消息交给所有绑定到交换机的队列(Publish/Subscribe模式)
b)
Direct
:定向,把消息交给符合指定routing key
的队列(Routing模式)
c)
Topic
:通配符,把消息交给符合routing pattern
(路由模式)的队列(Topics模式)
也就分别对应不同的工作模式
data:image/s3,"s3://crabby-images/1a0b6/1a0b677a6e69865b6c3970fc38c22a196e01941e" alt=""
1、引入依赖
java
<dependencies>
<!-- https://mvnrepository.com/artifact/com.rabbitmq/amqp-client -->
<dependency>
<groupId>com.rabbitmq</groupId>
<artifactId>amqp-client</artifactId>
<version>5.20.0</version>
</dependency>
</dependencies>
2、编写配置类
java
package rabbitmq.constant;
public class Constants {
public static final String HOST = "123.57.16.61";
public static final Integer PORT = 5672;
public static final String USERNAME = "study";
public static final String PASSWORD = "study";
public static final String VIRTUAL_HOST = "bite";
//发布订阅模式
public static final String FANOUT_EXCHANGE = "fanout.exchange";
public static final String FANOUT_QUEUE1 = "fanout.queue1";
public static final String FANOUT_QUEUE2 = "fanout.queue2";
}
3、编写生产者代码
java
package rabbitmq.fanout;
import com.rabbitmq.client.BuiltinExchangeType;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import rabbitmq.constant.Constants;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Producer {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory connectionFactory = new ConnectionFactory();
connectionFactory.setHost(Constants.HOST);
connectionFactory.setPort(Constants.PORT);
connectionFactory.setUsername(Constants.USERNAME);
connectionFactory.setPassword(Constants.PASSWORD);
connectionFactory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = connectionFactory.newConnection();
Channel channel = connection.createChannel();
//声明交换机
/**
* 交换机名称,交换机类型,开启可持久化(关机数据不会丢失)
*/
channel.exchangeDeclare(Constants.FANOUT_EXCHANGE, BuiltinExchangeType.FANOUT, true);
//声明队列
//queueDeclare 队列声明
channel.queueDeclare(Constants.FANOUT_QUEUE1, true, false, false, null);
channel.queueDeclare(Constants.FANOUT_QUEUE2, true, false, false, null);
//交换机和队列进行绑定
channel.queueBind(Constants.FANOUT_QUEUE1, Constants.FANOUT_EXCHANGE, "");
channel.queueBind(Constants.FANOUT_QUEUE2, Constants.FANOUT_EXCHANGE, "");
//发布消息
String msg = "hello fanout...";
//basicPublish (基础发布)
channel.basicPublish(Constants.FANOUT_EXCHANGE, "", null, msg.getBytes());
System.out.println("消息发送成功!!!");
//关闭资源
channel.close();
connection.close();
}
}
点击运行:
data:image/s3,"s3://crabby-images/551aa/551aaac02c941ac606745c8159c4a061b970fb38" alt=""
data:image/s3,"s3://crabby-images/660e7/660e742fa04469eb1f8d30e99f806f5dc94dddf3" alt=""
data:image/s3,"s3://crabby-images/74f5b/74f5b90220ef0ac8c23e4924834033bcca9f7627" alt=""
4、编写消费者代码
消费者1:
java
package rabbitmq.fanout;
import com.rabbitmq.client.*;
import rabbitmq.constant.Constants;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Consumer1 {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory connectionFactory = new ConnectionFactory();
connectionFactory.setHost(Constants.HOST);
connectionFactory.setPort(Constants.PORT);
connectionFactory.setUsername(Constants.USERNAME);
connectionFactory.setPassword(Constants.PASSWORD);
connectionFactory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = connectionFactory.newConnection();
Channel channel = connection.createChannel();
//声明队列
//queueDeclare 队列声明 (也可以省略)
channel.queueDeclare(Constants.FANOUT_QUEUE1, true, false, false, null);
//消费消息
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.FANOUT_QUEUE1,true,consumer);
}
}
消费者2:
java
package rabbitmq.fanout;
import com.rabbitmq.client.*;
import rabbitmq.constant.Constants;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Consumer2 {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory connectionFactory = new ConnectionFactory();
connectionFactory.setHost(Constants.HOST);
connectionFactory.setPort(Constants.PORT);
connectionFactory.setUsername(Constants.USERNAME);
connectionFactory.setPassword(Constants.PASSWORD);
connectionFactory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = connectionFactory.newConnection();
Channel channel = connection.createChannel();
//声明队列
//queueDeclare 队列声明 (也可以省略)
channel.queueDeclare(Constants.FANOUT_QUEUE2, true, false, false, null);
//消费消息
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.FANOUT_QUEUE2,true,consumer);
}
}
data:image/s3,"s3://crabby-images/b887a/b887ab7aae308d618002873eec52770ae56c2119" alt=""
二、Routing (路由模式)
队列和交换机的绑定, 不能是任意的绑定了, 而是要指定⼀个
BindingKey(RoutingKey的⼀种)
消息的发送方在向
Exchange
发送消息时, 也需要指定消息的RoutingKey
Exchange
也不再把消息交给每⼀个绑定的key, 而是根据消息的RoutingKey
进行判断, 只有队列绑定时的BindingKey
和发送消息的RoutingKey
完全⼀致, 才会接收到消息
data:image/s3,"s3://crabby-images/6f60f/6f60f42ad3551b0301f1f88125984ba9c1eaf322" alt=""
1、引入依赖
java
<dependencies>
<!-- https://mvnrepository.com/artifact/com.rabbitmq/amqp-client -->
<dependency>
<groupId>com.rabbitmq</groupId>
<artifactId>amqp-client</artifactId>
<version>5.20.0</version>
</dependency>
</dependencies>
2、编写配置类
java
package rabbitmq.constant;
public class Constants {
public static final String HOST = "123.57.16.61";
public static final Integer PORT = 5672;
public static final String USERNAME = "study";
public static final String PASSWORD = "study";
public static final String VIRTUAL_HOST = "bite";
//路由模式
public static final String DIRECT_EXCHANGE = "direct.exchange";
public static final String DIRECT_QUEUE1 = "direct.queue1";
public static final String DIRECT_QUEUE2 = "direct.queue2";
}
3、编写生产者代码
java
package rabbitmq.direct;
import com.rabbitmq.client.BuiltinExchangeType;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import rabbitmq.constant.Constants;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Producer {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory connectionFactory = new ConnectionFactory();
connectionFactory.setHost(Constants.HOST);
connectionFactory.setPort(Constants.PORT);
connectionFactory.setUsername(Constants.USERNAME);
connectionFactory.setPassword(Constants.PASSWORD);
connectionFactory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = connectionFactory.newConnection();
Channel channel = connection.createChannel();
//声明交换机
/**
* 交换机名称,交换机类型,开启可持久化(关机数据不会丢失)
*/
channel.exchangeDeclare(Constants.DIRECT_EXCHANGE, BuiltinExchangeType.DIRECT, true);
//声明队列
//queueDeclare 队列声明
channel.queueDeclare(Constants.DIRECT_QUEUE1, true, false, false, null);
channel.queueDeclare(Constants.DIRECT_QUEUE2, true, false, false, null);
//交换机和队列进行绑定
channel.queueBind(Constants.DIRECT_QUEUE1, Constants.DIRECT_EXCHANGE, "a");
channel.queueBind(Constants.DIRECT_QUEUE2, Constants.DIRECT_EXCHANGE, "a");
channel.queueBind(Constants.DIRECT_QUEUE2, Constants.DIRECT_EXCHANGE, "b");
channel.queueBind(Constants.DIRECT_QUEUE2, Constants.DIRECT_EXCHANGE, "c");
//发布消息
String msg_a = "hello direct, my routingkey is a...";
//basicPublish (基础发布)
channel.basicPublish(Constants.DIRECT_EXCHANGE, "", null, msg_a.getBytes());
String msg_b = "hello direct, my routingkey is b...";
channel.basicPublish(Constants.DIRECT_EXCHANGE, "", null, msg_b.getBytes());
String msg_c = "hello direct, my routingkey is c...";
channel.basicPublish(Constants.DIRECT_EXCHANGE, "", null, msg_c.getBytes());
System.out.println("消息发送成功!!!");
//关闭资源
channel.close();
connection.close();
}
}
data:image/s3,"s3://crabby-images/e6189/e6189c976179e1635b89ed367ec98171423f7390" alt=""
data:image/s3,"s3://crabby-images/160b3/160b36fca2ded5c25ba5406276c851b9c05fdcbd" alt=""
data:image/s3,"s3://crabby-images/be7df/be7dfdaebd78a41515425236ba53fe54b84df9e9" alt=""
data:image/s3,"s3://crabby-images/e03fd/e03fdc3e03e2853f8b0391381ad37a271fde2c84" alt=""
data:image/s3,"s3://crabby-images/18969/189696e62420aa8bb32eb83539bdc9095d01fd49" alt=""
4、编写消费者代码
消费者1:
java
package rabbitmq.direct;
import com.rabbitmq.client.*;
import rabbitmq.constant.Constants;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Consumer1 {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory connectionFactory = new ConnectionFactory();
connectionFactory.setHost(Constants.HOST);
connectionFactory.setPort(Constants.PORT);
connectionFactory.setUsername(Constants.USERNAME);
connectionFactory.setPassword(Constants.PASSWORD);
connectionFactory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = connectionFactory.newConnection();
Channel channel = connection.createChannel();
//声明队列
//queueDeclare 队列声明 (也可以省略)
channel.queueDeclare(Constants.DIRECT_QUEUE1, true, false, false, null);
//消费消息
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.DIRECT_QUEUE1,true,consumer);
}
}
data:image/s3,"s3://crabby-images/b2ab9/b2ab9abfd057421e232b6d25d6b144f20309005b" alt=""
消费者2:
java
package rabbitmq.direct;
import com.rabbitmq.client.*;
import rabbitmq.constant.Constants;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Consumer2 {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory connectionFactory = new ConnectionFactory();
connectionFactory.setHost(Constants.HOST);
connectionFactory.setPort(Constants.PORT);
connectionFactory.setUsername(Constants.USERNAME);
connectionFactory.setPassword(Constants.PASSWORD);
connectionFactory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = connectionFactory.newConnection();
Channel channel = connection.createChannel();
//声明队列
//queueDeclare 队列声明 (也可以省略)
channel.queueDeclare(Constants.DIRECT_QUEUE2, true, false, false, null);
//消费消息
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.DIRECT_QUEUE2,true,consumer);
}
}
data:image/s3,"s3://crabby-images/a9fa4/a9fa4347b308581d1d494415b097559e169d7790" alt=""
三、Topics (通配符模式)
Topics 和Routing模式的区别是:
- topics 模式使用的交换机类型为topic(Routing模式使用的交换机类型为direct)
- topic 类型的交换机在匹配规则上进行了扩展, Binding Key⽀持通配符匹配(direct类型的交换机路由规则是BindingKey和RoutingKey完全匹配)
data:image/s3,"s3://crabby-images/5dd6e/5dd6e4eccdf015e12560cc9852b494e2dd35bece" alt=""
在topic类型的交换机在匹配规则上, 有些要求:
- RoutingKey 是⼀系列由点( . )分隔的单词, ⽐如 " stock.usd.nyse ", " nyse.vmw ",
" quick.orange.rabbit " - BindingKey 和RoutingKey⼀样, 也是点( . )分割的字符串.
- Binding Key中可以存在两种特殊字符串, 用于模糊匹配
* 表⽰⼀个单词 # 表⽰多个单词(0-N个)
比如:
• Binding Key 为"d.a.b" 会同时路由到Q1 和Q2
• Binding Key 为"d.a.f" 会路由到Q1
• Binding Key 为"c.e.f" 会路由到Q2
• Binding Key 为"d.b.f" 会被丢弃, 或者返回给⽣产者(需要设置mandatory参数)
1、引入依赖
java
<dependencies>
<!-- https://mvnrepository.com/artifact/com.rabbitmq/amqp-client -->
<dependency>
<groupId>com.rabbitmq</groupId>
<artifactId>amqp-client</artifactId>
<version>5.20.0</version>
</dependency>
</dependencies>
2、编写配置类
java
package rabbitmq.constant;
public class Constants {
public static final String HOST = "123.57.16.61";
public static final Integer PORT = 5672;
public static final String USERNAME = "study";
public static final String PASSWORD = "study";
public static final String VIRTUAL_HOST = "bite";
//通配符模式
public static final String TOPIC_EXCHANGE = "topic.exchange";
public static final String TOPIC_QUEUE1 = "topic.queue1";
public static final String TOPIC_QUEUE2 = "topic.queue2";
3、编写生产者代码
java
package rabbitmq.topic;
import com.rabbitmq.client.BuiltinExchangeType;
import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import rabbitmq.constant.Constants;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Producer {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory connectionFactory = new ConnectionFactory();
connectionFactory.setHost(Constants.HOST);
connectionFactory.setPort(Constants.PORT);
connectionFactory.setUsername(Constants.USERNAME);
connectionFactory.setPassword(Constants.PASSWORD);
connectionFactory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = connectionFactory.newConnection();
Channel channel = connection.createChannel();
//声明交换机
/**
* 交换机名称,交换机类型,开启可持久化(关机数据不会丢失)
*/
channel.exchangeDeclare(Constants.TOPIC_EXCHANGE, BuiltinExchangeType.TOPIC, true);
//声明队列
//queueDeclare 队列声明
channel.queueDeclare(Constants.TOPIC_QUEUE1, true, false, false, null);
channel.queueDeclare(Constants.TOPIC_QUEUE2, true, false, false, null);
//交换机和队列进行绑定
channel.queueBind(Constants.TOPIC_QUEUE1, Constants.TOPIC_EXCHANGE, "*.a.*");
channel.queueBind(Constants.TOPIC_QUEUE2, Constants.TOPIC_EXCHANGE, "*.*.b");
channel.queueBind(Constants.TOPIC_QUEUE2, Constants.TOPIC_EXCHANGE, "c.#");
//发布消息
String msg_a = "hello topic, my routingkey is ae.a.f...";
//basicPublish (基础发布)
channel.basicPublish(Constants.TOPIC_EXCHANGE, "ae.a.f", null, msg_a.getBytes());
String msg_b = "hello topic, my routingkey is ef.a.b...";
channel.basicPublish(Constants.TOPIC_EXCHANGE, "ef.a.b", null, msg_b.getBytes());
String msg_c = "hello topic, my routingkey is c.ef.b...";
channel.basicPublish(Constants.TOPIC_EXCHANGE, "c.ef.b", null, msg_c.getBytes());
System.out.println("消息发送成功!!!");
//关闭资源
channel.close();
connection.close();
}
}
4、编写消费者代码
消费者1:
java
package rabbitmq.topic;
import com.rabbitmq.client.*;
import rabbitmq.constant.Constants;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Consumer1 {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory connectionFactory = new ConnectionFactory();
connectionFactory.setHost(Constants.HOST);
connectionFactory.setPort(Constants.PORT);
connectionFactory.setUsername(Constants.USERNAME);
connectionFactory.setPassword(Constants.PASSWORD);
connectionFactory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = connectionFactory.newConnection();
Channel channel = connection.createChannel();
//声明队列
//queueDeclare 队列声明 (也可以省略)
channel.queueDeclare(Constants.TOPIC_QUEUE1, true, false, false, null);
//消费消息
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.TOPIC_QUEUE1,true,consumer);
}
}
data:image/s3,"s3://crabby-images/c29d7/c29d7997e44df7d370cdabffd24c1eb719a61efe" alt=""
消费者2:
java
package rabbitmq.topic;
import com.rabbitmq.client.*;
import rabbitmq.constant.Constants;
import java.io.IOException;
import java.util.concurrent.TimeoutException;
public class Consumer2 {
public static void main(String[] args) throws IOException, TimeoutException {
ConnectionFactory connectionFactory = new ConnectionFactory();
connectionFactory.setHost(Constants.HOST);
connectionFactory.setPort(Constants.PORT);
connectionFactory.setUsername(Constants.USERNAME);
connectionFactory.setPassword(Constants.PASSWORD);
connectionFactory.setVirtualHost(Constants.VIRTUAL_HOST);
Connection connection = connectionFactory.newConnection();
Channel channel = connection.createChannel();
//声明队列
//queueDeclare 队列声明 (也可以省略)
channel.queueDeclare(Constants.TOPIC_QUEUE2, true, false, false, null);
//消费消息
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.TOPIC_QUEUE2,true,consumer);
}
}
data:image/s3,"s3://crabby-images/05ead/05ead3d17db21aad5647fc2eda4a7dcfe51d023f" alt=""
data:image/s3,"s3://crabby-images/8d600/8d600ca272fbbe6f7bf0a3fd930638f9b1ec453e" alt=""
data:image/s3,"s3://crabby-images/44af2/44af2046549788e3071c8a76fe0ebf51fd12b838" alt=""