RabbitMQ - Spring配置 - topic《通配符模式》

 SpringMain_Topic.java

package com.xzp.rabbitmq.spring.topic;

import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.context.support.AbstractApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

/**
 * main程序
 * 交换机为- 》topic:  代表 "通配符模式"《可选择性接收消息, 采用通配符模式 * 符号代表1个, # 符号代表一个或多个》
 */
public class SpringMain_Topic {

    public static void main(final String... args) throws Exception {
        AbstractApplicationContext ctx = new ClassPathXmlApplicationContext("classpath:spring/rabbitmq-context-fanout.xml");
        //RabbitMQ模板
        //RabbitTemplate template = ctx.getBean(RabbitTemplate.class);
        RabbitTemplate amqpTemplate01 = (RabbitTemplate) ctx.getBean("amqpTemplate_fanout");
        RabbitTemplate amqpTemplate02 = (RabbitTemplate) ctx.getBean("amqpTemplate_direct");
        RabbitTemplate amqpTemplate03 = (RabbitTemplate) ctx.getBean("amqpTemplate_topic");

        //发送消息
        String str01 = "Hello, world!, This exchange is fanout !!";
        amqpTemplate01.convertAndSend(str01);
        System.out.println();
        System.out.println("发送消息: " + str01);
        System.out.println();

        //发送消息
        String str02 = "Hello, world!, This exchange is direct !!";
        amqpTemplate02.convertAndSend(str02);
        System.out.println();
        System.out.println("发送消息: " + str02);
        System.out.println();

        //发送消息
        String str03 = "Hello, world!, This exchange is topic !!";
        amqpTemplate03.convertAndSend(str03);
        System.out.println();
        System.out.println("发送消息: " + str03);
        System.out.println();


        // 休眠1秒
        Thread.sleep(1000);
        //容器销毁
        ctx.destroy();
    }
}

 Customer01.java

package com.xzp.rabbitmq.spring.topic;

/**
 * 消费者1
 * 交换机为- 》topic:  代表 "通配符模式"《可选择性接收消息, 采用通配符模式 * 符号代表1个, # 符号代表一个或多个》
 */
public class Customer01 {

    //具体执行业务的方法
    public void listen(String foo) {

        System.out.println();
        System.out.println();
        System.out.println("topic 消费者 Customer01: " + foo);
        System.out.println();
        System.out.println();

    }
}

Customer02.java

package com.xzp.rabbitmq.spring.topic;

/**
 * 消费者2
 * 交换机为- 》topic:  代表 "通配符模式"《可选择性接收消息, 采用通配符模式 * 符号代表1个, # 符号代表一个或多个》
 */
public class Customer02 {

    //具体执行业务的方法
    public void listen(String foo) {

        System.out.println();
        System.out.println();
        System.out.println("topic 消费者 Customer02: " + foo);
        System.out.println();
        System.out.println();

    }
}

rabbitmq-context-fanout.xml

<beans xmlns="http://www.springframework.org/schema/beans"
      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:rabbit="http://www.springframework.org/schema/rabbit"
      xsi:schemaLocation="http://www.springframework.org/schema/rabbit
   http://www.springframework.org/schema/rabbit/spring-rabbit-1.4.xsd
   http://www.springframework.org/schema/beans
   http://www.springframework.org/schema/beans/spring-beans-4.2.xsd">

   <!--定义连接工厂-->
   <rabbit:connection-factory id="connectionFactory" host="127.0.0.1" port="5672" username="baseup" password="wukong@123" virtual-host="/baseup"  />

   <!-- MQ的管理,包括队列、交换器等 -->
   <rabbit:admin id="rabbitAdmin_fanout" connection-factory="connectionFactory" />
   <rabbit:admin id="rabbitAdmin_direct" connection-factory="connectionFactory" />
   <rabbit:admin id="rabbitAdmin_topic" connection-factory="connectionFactory" />

   <!-- 定义Rabbit模板,指定连接工厂以及定义exchange -->
   <rabbit:template id="amqpTemplate_fanout" connection-factory="connectionFactory" exchange="exchange_setting_fanout" />
   <rabbit:template id="amqpTemplate_direct" connection-factory="connectionFactory" exchange="exchange_setting_direct" routing-key="wh.public" />
   <rabbit:template id="amqpTemplate_topic"  connection-factory="connectionFactory" exchange="exchange_setting_topic" routing-key="wh.public.news" />


   <!-- 定义队列,自动声明 -->
   <rabbit:queue name="exchange_setting_fanout_queue01" auto-declare="true" declared-by="rabbitAdmin_fanout" />
   <rabbit:queue name="exchange_setting_fanout_queue02" auto-declare="true" declared-by="rabbitAdmin_fanout" />

   <rabbit:queue name="exchange_setting_direct_queue01" auto-declare="true" declared-by="rabbitAdmin_direct" />
   <rabbit:queue name="exchange_setting_direct_queue02" auto-declare="true" declared-by="rabbitAdmin_direct" />

   <rabbit:queue name="exchange_setting_topic_queue01" auto-declare="true" declared-by="rabbitAdmin_topic" />
   <rabbit:queue name="exchange_setting_topic_queue02" auto-declare="true" declared-by="rabbitAdmin_topic" />

   <!-- 定义交换器,自动声明 -->
   <rabbit:fanout-exchange name="exchange_setting_fanout" auto-declare="true" declared-by="rabbitAdmin_fanout">
      <rabbit:bindings>
         <rabbit:binding queue="exchange_setting_fanout_queue01" />
         <rabbit:binding queue="exchange_setting_fanout_queue02" />
      </rabbit:bindings>
   </rabbit:fanout-exchange>

   <rabbit:direct-exchange name="exchange_setting_direct" auto-declare="true" declared-by="rabbitAdmin_direct">
      <rabbit:bindings>
         <rabbit:binding queue="exchange_setting_direct_queue01" key="wh.public" />
         <rabbit:binding queue="exchange_setting_direct_queue02" key="wh.public" />
      </rabbit:bindings>
   </rabbit:direct-exchange>

   <rabbit:topic-exchange name="exchange_setting_topic" auto-declare="true" declared-by="rabbitAdmin_topic">
      <rabbit:bindings>
         <rabbit:binding queue="exchange_setting_topic_queue01" pattern="wh.*.news" />
         <rabbit:binding queue="exchange_setting_topic_queue02" pattern="wh.#" />
      </rabbit:bindings>
   </rabbit:topic-exchange>

   <!-- 队列监听 -->
   <rabbit:listener-container connection-factory="connectionFactory">
      <rabbit:listener ref="consumer01" method="listen" queue-names="exchange_setting_fanout_queue01" />
      <rabbit:listener ref="consumer02" method="listen" queue-names="exchange_setting_fanout_queue02" />
   </rabbit:listener-container>


   <rabbit:listener-container connection-factory="connectionFactory">
      <rabbit:listener ref="user01" method="listen" queue-names="exchange_setting_direct_queue01" />
      <rabbit:listener ref="user02" method="listen" queue-names="exchange_setting_direct_queue02"  />
   </rabbit:listener-container>

   <rabbit:listener-container connection-factory="connectionFactory">
      <rabbit:listener ref="customer01" method="listen" queue-names="exchange_setting_topic_queue01" />
      <rabbit:listener ref="customer02" method="listen" queue-names="exchange_setting_topic_queue02"  />
   </rabbit:listener-container>

   <!--定义消费Bean-->
   <bean id="consumer01" class="com.xzp.rabbitmq.spring.fanout.Consumer01" />
   <bean id="consumer02" class="com.xzp.rabbitmq.spring.fanout.Consumer02" />

   <bean id="user01" class="com.xzp.rabbitmq.spring.direct.User01" />
   <bean id="user02" class="com.xzp.rabbitmq.spring.direct.User02" />

   <bean id="customer01" class="com.xzp.rabbitmq.spring.topic.Customer01" />
   <bean id="customer02" class="com.xzp.rabbitmq.spring.topic.Customer02" />

</beans>

### RabbitMQTopic 模式的使用与配置 #### 1. 基本概念 在 RabbitMQTopic 模式下,消息路由键(routing key)和绑定键(binding key)都可以包含多个单词,这些单词之间用点号 `.` 分隔。生产者发送消息时指定一个路由键,而消费者则通过定义特定的绑定键来订阅感兴趣的消息类型。 #### 2. 生产者的实现方式 对于生产者而言,在发送消息到 Topic 类型的交换机时,只需要调用 `convertAndSend` 方法并传入相应的参数即可。例如: ```java @Autowired private RabbitTemplate rabbitTemplate; public void sendMsg() { String exchangeName = "topicExchange"; String routingKey = "key1"; // 路由键应遵循一定的命名规则以便于匹配 Object messageBody = "This is a test message."; rabbitTemplate.convertAndSend(exchangeName, routingKey, messageBody); } ``` 此段代码展示了如何利用 Spring AMQP 提供的 `RabbitTemplate` 来向名为 `"topicExchange"` 的 Topic Exchange 发送一条带有自定义路由键的消息[^2]。 #### 3. 绑定关系设置 为了使消费者能够接收到符合条件的消息,需建立合适的绑定关系。这通常是在声明队列之后完成的操作,即先创建好队列再将其与某个 Topic Exchange 关联起来,并指明具体的 binding key 形式。比如下面的例子说明了一个简单的绑定过程: ```java // 定义队列名称以及对应的 Binding Key 表达式 String queueName = "testQueue"; String pattern = "*.orange.*"; // 执行绑定操作 amqpAdmin.declareBinding(BindingBuilder.bind(new Queue(queueName)) .to(new DirectExchange("topicExchange")) .with(pattern)); ``` 这里使用的 `pattern` 参数支持通配符语法,其中星号(*)代表任意单个词,井号(#)表示零个或多个词语[^4]。 #### 4. 实际应用场景中的注意事项 当实际应用中涉及到复杂的业务逻辑时,建议合理规划路由键的设计原则,确保其既具有足够的表达力又不会过于复杂难以维护;同时也要注意性能方面的影响因素,如过多的队列可能会导致资源消耗增加等问题[^3]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

suenpeng

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值