Spring AMQP,以及各种模式的使用

本文详细介绍了Spring AMQP的使用,包括AMQP协议的基础知识、Spring AMQP的API规范及其与RabbitMQ的结合。文章通过实例演示了Spring AMQP支持的简单工作模式、工作队列模式、发布订阅模式、路由模式和通配符模式,展示了如何配置、发送和消费消息。

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

安装RabbitMQ教程:点击,查看安装教程

什么是AMQP?

amqp(高级消息队列协议)是一种协议,类似http协议

amqp协议规定了,消息生产者和mq及消息消费者和mq通信的标准

什么是Spring AMQP?

        Spring基于amqp协议定义的一套API规范(spring-amqp),并且由rabbitmq官方对这一套规范进行了实现(spring-rabbit);

        SpringAMQP是基于RabbitMQ封装的一套模板,并且还利用SpringBoot对其实现了自动装配,使用起来非常方便。

SpringAMQP提供了三个功能

  • 自动声明队列、交换机及其绑定关系

  • 基于注解的监听器模式,异步接收消息

  • 封装了RabbitTemplate工具,用于发送消息

工作模式

1.简单工作模式

2.工作队列模式

3.发布订阅模式

4.路由模式

5.通配符模式

模式使用如下:

1.简单工作模式

工作流程图:

        模式特点:生产者发送消息到默认交换机,默认交换机根据消息队列的名称,将消息转发到指定队列。

        消费者监听消息队列,获取消息。

项目结构

 1.父工程的pom.xml

<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>2.3.9.RELEASE</version>
    <relativePath/>
</parent>
<properties>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
    <java.version>1.8</java.version>
</properties>

<dependencies>
    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
    </dependency>
</dependencies>

2. 消息生产者模块

        2.1.消息生产者模块pom.xml,application.yml配置,并编写启动类

<dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
        <scope>test</scope>
    </dependency>
    <!--AMQP依赖,包含RabbitMQ-->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-amqp</artifactId>
    </dependency>
</dependencies>

application.yml配置

server:
  port: 8081
spring:
  rabbitmq:
    # 主机名
    host: 192.168.220.131
    # 端口
    port: 5672
    # 用户名和密码
    username: guest
    password: guest
    # 虚拟主机
    virtual-host: /

        2.2.编写消息生产者的Controller方法,当访问sendMessage的方法时,向消息队列中发送hello world

@RestController
@RequestMapping("/producer")
public class ProducerController {

    @Autowired
    private RabbitTemplate rabbitTemplate;

    @GetMapping("/sendMessage")
    public String sendMessage() {
        //RabbitTemplate实现消息发送
        //参数一:默认的交换机空字符串
        //参数二:路由到那个消息队列
        //参数三:发送的消息内容
        rabbitTemplate.convertAndSend("","simple.queue","hello world");
        return "success";
    }
}

        2.3.启动生产者服务,查看RabbitMQ中是否把消息存储到队列中,以及访问Controller方法是否成功

 3.消息消费者

        3.1.消息消费者模块pom.xml,application.yml配置,并编写启动类

代码和消息生产者的一样,但是application.yml记得修改一下端口

        3.2.编写消息消费者的监听,监听消息队列,获取消息

@Component
@Slf4j
public class SimpleQueueLisener {
    //监听那个队列
    @RabbitListener(queues = "simple.queue")
    public void receiveMessage(String message) {
        //由于对消息不做如何操作,在日志打印一下
        log.info("接受到的消息是--->{}",message);
    }
}

        2.3.启动消费者服务,查看RabbitMQ中消息是否被取出,打印到控制台


2.工作队列模式

工作流程图:

        在简单模式的项目结构上,多启动一个消息消费者的主启动类。多个消费者去监听同一个消息队列。

        当消息队列数据过多时,一个消费者无法跟上生产者的效率,很可能出现故障,如果还有一个消费者同时去抢消息,那么效率就会很快。


3.发布订阅模式(Fanout:广播,将消息交给所有绑定到交换机的队列 )

工作流程图:

1.生产者发送消息到某个交换机(不能是默认交换机)

2.由交换机将消息转发给绑定它的所有消息队列

3.消费者可以从每一个消息队列中获取消息

1.项目结构同上一样

2.消息生产者模块(该配置类要在消息消费者来一份,由于部署不知道先是生产者还是消费者,防止消费者启动没有交换机和消息队列)

        2.1.编写一个Config类,创建交换机,消息队列,并绑定交换机和消息队列

//配置类
@Configuration
public class FanoutConfig {
    /**
     * 创建交换机
     */
    @Bean
    public Exchange createFanoutExchange(){
        //参数一:交换机名称
        //参数二:是否支持持久化
        //参数三:是否自动删除
        return new FanoutExchange("exchange",true,false);
        //构建者模式创建交换机
        //return ExchangeBuilder.fanoutExchange("exchange").durable(true).build();
    }

    /**
     *创建队列fqueue1
     */
    @Bean
    public Queue createFQueue1(){
        return QueueBuilder.durable("fqueue1").build();
        //队列名字,是否支持持久化
        //return new Queue("fqueue1",true);
    }

    /**
     *创建队列fqueue2
     */
    @Bean
    public Queue createFQueue2(){
        return QueueBuilder.durable("fqueue2").build();
        //队列名字,是否支持持久化
        //return new Queue("fqueue1",true);
    }

    /**
     *绑定交换机和队列
     */
    public Binding bindFqueuelToExchange(@Qualifier("createFanoutExchange") Exchange exchange,
                                         @Qualifier("createFQueue1") Queue queue){
        //将队列1于交换机绑定,with是路由规则是绑定所有
        return BindingBuilder.bind(queue).to(exchange).with("").noargs();
    }

    /**
     *绑定交换机和队列
     */
    public Binding bindFqueue2ToExchange(@Qualifier("createFanoutExchange") Exchange exchange,
                                         @Qualifier("createFQueue2") Queue queue){
        //将队列2于交换机绑定,with是路由规则是绑定所有
        return BindingBuilder.bind(queue).to(exchange).with("").noargs();
    }

}

        2.2.编写Controller方法向交换机发送消息

@GetMapping("/hh")
public String sendMm(){
    //将消息发送到exchange交换机,路由:路由到所有,默认为"",消息是:你好呀!
    rabbitTemplate.convertAndSend("exchange","","你好呀!");
    return "OK";
}

3.消息消费者模块

        3.1.创建一个C1模块,编写监听,监听消息

@Component
@Slf4j
public class C1Lisener {
    //监听那个队列
    @RabbitListener(queues = "fqueue1")
    public void receiveMessage(String message) {
        //由于对消息不做如何操作,在日志打印一下
        log.info("接受到的消息是--->{}",message);
    }
}

        3.2.创建一个C2模块,编写监听,监听消息

@Component
@Slf4j
public class C1Lisener {
    //监听那个队列
    @RabbitListener(queues = "fqueue2")
    public void receiveMessage(String message) {
        //由于对消息不做如何操作,在日志打印一下
        log.info("接受到的消息是--->{}",message);
    }
}

4.查看控制台,消息是否被监听到


4.路由模式(directExchange)

工作流程图:

1.生产者发送消息时,需指定交换机名称(directExchange),还要指定路由规则,然后发送到交换机

2.交换机事先通过不同的路由规则和消息队列进行绑定,当交换机拿到消息后,通过路由规则路由到指定队列

3.消费者通过监听获取消息

1.项目结构同上一样

2.消息生产者模块(该配置类要在消息消费者来一份,由于部署不知道先是生产者还是消费者,防止消费者启动没有交换机和消息队列)

        2.1.编写一个Config类,创建交换机,消息队列,并绑定交换机和消息队列

@Configuration
public class DirectConfig {
    /**
     * 创建交换机
     */
    @Bean
    public Exchange createDirect(){
        return ExchangeBuilder.directExchange("dirct.exchange").durable(true).build();
    }

    /**
     * 创建队列
     */
    @Bean
    public Queue Dqueue1(){
        return QueueBuilder.durable("d.queue1").build();
    }

    /**
     * 创建队列
     */
    @Bean
    public Queue Dqueue2(){
        return QueueBuilder.durable("d.queue2").build();
    }

    /**
     * 绑定
     */
    @Bean
    public Binding bindDQueue1ToExchange(@Qualifier("createDirect") Exchange exchange,
                                         @Qualifier("Dqueue1") Queue queue){
        return BindingBuilder.bind(queue).to(exchange).with("mm").noargs();
    }

    /**
     * 绑定
     */
    @Bean
    public Binding bindDQueue2ToExchange(@Qualifier("createDirect") Exchange exchange,
                                         @Qualifier("Dqueue2") Queue queue){
        return BindingBuilder.bind(queue).to(exchange).with("gg").noargs();
    }
}

        2.2.编写Controller方法向交换机发送消息

@GetMapping("/mm")
public String sendMm(){
    rabbitTemplate.convertAndSend("dirct.exchange","mm","大美女");
    return "OK";
}

@GetMapping("/gg")
public String sendDm(){
    rabbitTemplate.convertAndSend("dirct.exchange","gg","大哥哥");
    return "OK OK";

3.消息消费者模块

        3.1.创建一个C1模块,编写监听,监听消息

@Component
@Slf4j
public class C1Lisener {
    //监听那个队列
    @RabbitListener(queues = "d.queue1")
    public void receiveMessage(String message) {
        //由于对消息不做如何操作,在日志打印一下
        log.info("接受到的消息是--->{}",message);
    }
}

        3.2.创建一个C2模块,编写监听,监听消息

@Component
@Slf4j
public class C2Lisener {
    //监听那个队列
    @RabbitListener(queues = "d.queue2")
    public void receiveMessage(String message) {
        //由于对消息不做如何操作,在日志打印一下
        log.info("接受到的消息是--->{}",message);
    }
}

4.查看控制台,消息是否被监听到

 

 


5.通配符模式(topicExchange),和路由模式非常像,路由规则不同

工作流程图:

1.生产者发送消息时,需指定交换机名称(topicExchange),还要指定路由规则,然后发送到交换机

        通配模式的路由规则:

        #:统配0~多个单词,单词之间用.分隔

        *:统配一个任意单词

2.交换机事先通过不同的路由规则和消息队列进行绑定,当交换机拿到消息后,通过路由规则路由到指定队列

3.消费者通过监听获取消息

1.项目结构同上一样

2.消息生产者模块(该配置类要在消息消费者来一份,由于部署不知道先是生产者还是消费者,防止消费者启动没有交换机和消息队列)

        2.1.编写一个Config类,创建交换机,消息队列,并绑定交换机和消息队列

@Configuration
public class TopicConfig {
    /**
     * 创建交换机
     */
    @Bean
    public Exchange topicExchange(){
        return ExchangeBuilder.topicExchange("topic.exchange").durable(true).build();
    }

    /**
     * 创建消息队列
     */
    @Bean
    public Queue queue1(){
        return QueueBuilder.durable("t.queue1").build();
    }

    @Bean
    public Queue queue2(){
        return QueueBuilder.durable("t.queue2").build();
    }

    /**
     * 绑定交换机和消息队列
     */
    @Bean
    public Binding bindQ1ToTopic(@Qualifier("topicExchange") Exchange exchange,
                                 @Qualifier("queue1") Queue queue){
        return BindingBuilder.bind(queue).to(exchange).with("*.gao.*").noargs();
    }

    /**
     * 绑定交换机和消息队列
     */
    @Bean
    public Binding bindQ2ToTopic(@Qualifier("topicExchange") Exchange exchange,
                                 @Qualifier("queue2") Queue queue){
        return BindingBuilder.bind(queue).to(exchange).with("da.#").noargs();
    }
}

        2.2.编写Controller方法向交换机发送消息

@GetMapping("/hh")
public String send1(){
    //指定交换机的名称
    //通配符模式的路由规则
    //消息内容
    rabbitTemplate.convertAndSend("topic.exchange","da.gao.aaa","这人牛逼");
    return "GLgoodMan";
}

@GetMapping("/dd")
public String send2(){
    rabbitTemplate.convertAndSend("topic.exchange","da.hh","hhhhh");
    return "hellow";
}

 3.消息消费者模块

        3.1.创建一个C2模块,编写监听,监听消息

@Component
@Slf4j
public class TopLisener {
    @RabbitListener(queues = "t.queue1")
    public void receiveMessage(String message){
        //由于对消息不做如何操作,在日志打印一下
        log.info("接受到的消息是--->{}",message);
    }
}

        3.2.创建一个C2模块,编写监听,监听消息

@Component
@Slf4j
public class TopLisener {
    @RabbitListener(queues = "t.queue2")
    public void receiveMessage(String message){
        //由于对消息不做如何操作,在日志打印一下
        log.info("接受到的消息是--->{}",message);
    }
}

4.查看控制台,消息是否被监听到

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值