RabbitMQ学习笔记六:Exchange的学习(4)

本文介绍了RabbitMQ中不常用的Headers类型的Exchange,它与Topic类型相似,通过匹配Header键值对而非routingKey来路由消息。讲解了"any"和"all"两种匹配规则,并通过生产者和消费者示例代码展示了如何使用Headers Exchange,强调在实际应用中,通常更倾向于使用direct类型的交换机。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、概述

    前面三篇文章讲述了RabbitMQ 常用的三种Exchange类型,这篇文章学习一下第四种不常用的Exchange类型:Headers这种类型与topic类型类似,只不过不是匹配routingKeys,是匹配AMQP协议中的Header,Header是一个HashTable类型的键值对,而routingKey是String类型的字符串。功能与Topic相同,消息发送者绑定消息的键值对,匹配交换机与队列之间绑定的键值对,匹配规则“x-match”有两种,一种是“any”,只要一组键值对匹配成功即可发送消息到该队列,另一种是“all”,即需要所有键值对都匹配才可以发送消息。

    大概的场景应用示意图如下,详细说明见示例代码:

    

二、源代码

    我们先测试any类型的headers,先写生产者代码,相关说明已在注释中标明

package com.cn.chenxyt.mq;

import java.io.IOException;
import java.util.Hashtable;
import java.util.Map;

import com.rabbitmq.client.AMQP.BasicProperties;  
import com.rabbitmq.client.AMQP.BasicProperties.Builder; 
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import com.rabbitmq.client.Channel;


public class MqProducer {
	 public final static String EXCHANGE_NAME="EX_HEADER";
	 public static void main(String[] args) throws IOException, InterruptedException {
		 //创建连接工厂
		 ConnectionFactory factory = new ConnectionFactory();
		 //设置主机、用户名、密码和客户端端口号
		 factory.setHost("localhost");
		 factory.setUsername("guest");
		 factory.setPassword("guest");
		 factory.setPort(5672);
		 //创建一个新的连接 即TCP连接
		 Connection connection = factory.newConnection();
		 //创建一个通道
		 Channel channel = connection.createChannel();
		 //创建一个交换机
		 channel.exchangeDeclare(EXCHANGE_NAME,"headers");
		 //定义发送消息的要绑定的键值对
		 Map<String,Object> headers =  new Hashtable<String, Object>();
		 headers.put("aaa", "111");
		 headers.put("bbb", "222");  
	     Builder properties = new BasicProperties.Builder();  
	     properties.headers(headers);
		 for(int i = 0;i<500;i++){
			 String message = "hello" + (i);
			 //发送消息 绑定header键值对
			 channel.basicPublish(EXCHANGE_NAME,"",properties.build(),message.getBytes());
			 System.out.println("发送消息:" + message);
			 Thread.sleep(2000);
		 }
	}
}

    消费者1代码,相关说明已经在注释中标明

package com.cn.chenxyt.mq;

import java.io.IOException;
import java.util.Hashtable;
import java.util.Map;

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import com.rabbitmq.client.Consumer;
import com.rabbitmq.client.DefaultConsumer;
import com.rabbitmq.client.Envelope;
import com.rabbitmq.client.AMQP.BasicProperties;
public class MqConsumer1 {
	 public final static String EXCHANGE_NAME="EX_HEADER";
	 public final static String QUEUE_NAME="queue1";
	 
	 
	 public static void main(String[] args) throws IOException, InterruptedException {
		 //创建连接工厂
		 ConnectionFactory factory = new ConnectionFactory();
		 //设置主机
		 factory.setHost("localhost");
		 //创建一个新的连接 即TCP连接
		 Connection connection = factory.newConnection();
		 //创建一个通道
		 Channel channel = connection.createChannel();
		 //声明队列
		 channel.queueDeclare(QUEUE_NAME, true, false, false, null);
		 //创建一个交换机
		 channel.exchangeDeclare(EXCHANGE_NAME,"headers");
		 //定义绑定规则
	     Map<String, Object> headers = new Hashtable<String, Object>(); 
	     //any 匹配任意一组即可 all 全部匹配
	     headers.put("x-match", "any");
	     headers.put("aaa", "111");  
	     headers.put("bbb", "222");
		 //绑定队列到交换机
		 channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "",headers);
	     System.out.println("Consumer1 Waiting Received messages");
	     //DefaultConsumer类实现了Consumer接口,通过传入一个channel,
	     //告诉服务器我们需要哪个channel的消息并监听channel,如果channel中有消息,就会执行回调函数handleDelivery
	     Consumer consumer = new DefaultConsumer(channel) {
	            @Override
	            public void handleDelivery(String consumerTag, Envelope envelope,
	                                       BasicProperties properties, byte[] body)
	                    throws IOException {
	                String message = new String(body, "UTF-8");
	                System.out.println("Consumer1 Received '" + message + "'");
	    	     //   int i = 1/0;
	            }
	        };
	        //自动回复队列应答 -- RabbitMQ中的消息确认机制
	        //false 不自动回复应答 
	        channel.basicConsume(QUEUE_NAME,true, consumer);
	}
}

    消费者2代码,与1基本相同,只不过新建个队列和绑定的header,有一点要说明一下,所有新测试的交换机类型,都需要把之前已经存在的同名的交换机或者同名的队列删除,不然的话新建不会生效,即使参数不同。

package com.cn.chenxyt.mq;

import java.io.IOException;
import java.util.Hashtable;
import java.util.Map;

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import com.rabbitmq.client.Consumer;
import com.rabbitmq.client.DefaultConsumer;
import com.rabbitmq.client.Envelope;
import com.rabbitmq.client.AMQP.BasicProperties;
public class MqConsumer2 {
	 public final static String EXCHANGE_NAME="EX_HEADER";
	 public final static String QUEUE_NAME="queue2";
	 public static void main(String[] args) throws IOException {
		 //创建连接工厂
		 ConnectionFactory factory = new ConnectionFactory();
		 //设置主机
		 factory.setHost("localhost");
		 //创建一个新的连接 即TCP连接
		 Connection connection = factory.newConnection();
		 //创建一个通道
		 Channel channel = connection.createChannel();
		 //声明队列
		 channel.queueDeclare(QUEUE_NAME, true, false, false, null);
		 //创建一个交换机
		 channel.exchangeDeclare(EXCHANGE_NAME,"headers");
		 //定义绑定规则
	     Map<String, Object> headers = new Hashtable<String, Object>(); 
	     //any 匹配任意一组即可 all 全部匹配
	     headers.put("x-match", "any");
	     headers.put("aaa", "111");  
	     headers.put("ccc", "333");
		 //绑定队列到交换机
		 channel.queueBind(QUEUE_NAME, EXCHANGE_NAME, "",headers);
	     System.out.println("Consumer2 Waiting Received messages");
	     //DefaultConsumer类实现了Consumer接口,通过传入一个channel,
	     //告诉服务器我们需要哪个channel的消息并监听channel,如果channel中有消息,就会执行回调函数handleDelivery
	     Consumer consumer = new DefaultConsumer(channel) {
	            @Override
	            public void handleDelivery(String consumerTag, Envelope envelope,
	                                       BasicProperties properties, byte[] body)
	                    throws IOException {
	                String message = new String(body, "UTF-8");
	                System.out.println("Consumer2 Received '" + message + "'");
	            }
	        };
	        //自动回复队列应答 -- RabbitMQ中的消息确认机制
	        channel.basicConsume(QUEUE_NAME, true, consumer);
	}
}

    分别启动消费者1和消费者2,使他们处在监听状态,可以看到管理台有新建的headers类型的交换机和两个队列

    

    

    

   

    

    启动生产者,可以看到消费者1和消费者2都收到了消息。说明any功能生效,即匹配到了任意一组键值对即可发送消息。

    

    

    

    接下来我们测试all的情况,在管理台删除已经存在的两条队列queue1、queue2,修改消费者1和消费者2中的any修改为all

    

	     //any 匹配任意一组即可 all 全部匹配
	     headers.put("x-match", "all");

    这时我们重新启动消费者1和消费者2使他们处于监听状态,可以在管理台看见绑定规则x-match变为all

    

    启动生产者,可以看见消费者1和消费者2都收不到消息了。

    

    

    

    修改生产者代码,新增一组键值对,保证与queue1绑定的headers键值对完全匹配

    

		 //定义发送消息的要绑定的键值对
		 Map<String,Object> headers =  new Hashtable<String, Object>();
	     headers.put("aaa", "111");  
	     headers.put("bbb", "222");

    重新启动生产者,可以看到消费者1收到了消息,消费者2没有收到消息,即all的匹配规则生效了。

    

    

    

    以上就是关于headers类型的exchange的应用示例,实际应用场景中,同类型的更偏向于使用direct类型的交换机。

三、代码下载

    代码下载地址:https://pan.baidu.com/s/1jJ5FOQq

    

    

   

    

    



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值