AMQP[RabbitMQ]小结

消息队列:

组成:

交换器,队列,绑定

作用:异步处理,削峰,服务解耦

交换器

RabbitMQ常见的exchange(交换器)类型:

  • direct--路由键完全匹配才可以

  • fanout--广播

  • topic --主题,模糊匹配路由键

队列

messagequeue:

组成:

  • 路由键 routine-key---决定消息发给谁

  • 优先级priority--决定消息发送的优先级

  • 分发模式deliver-mode--决定消息的发送方式--持久化等

绑定

binding:

依赖

复制代码
 <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-amqp</artifactId>
            <version>2.6.4</version>
        </dependency>

配置文件:

复制代码
spring:
  rabbitmq:
    username: zzy
    password: 1234
    host: 172.24.232.166

注册队列:

复制代码
@Configuration
public class RabbitConfig {
    @Bean
    protected Queue queue(){

        Queue queue= new Queue("zzy");
        return  queue;
    }
}

生产者发送消息

复制代码
//作为信息的发布者
@SpringBootTest(classes = ApplicationRabbitMq.class)
public class RabbitTest {
    //Amqp模板类
    @Autowired
    private AmqpTemplate amqpTemplate;

    @Test
    void RabbitTest(){
        amqpTemplate.convertAndSend("zzy","hello world");
        System.out.println("success");
    }
}

消费者消费消息:

复制代码
@Component  //加注解,不然无法解析
public class RabbitMqConsumer {

    @RabbitListener(queues = "zzy")//订阅的队列
    public void Listened1(String msg){

        System.out.println("取出的消息1----"+msg);
    }
    @RabbitListener(queues = "zzy")
    public void Listened2(String msg){

        System.out.println("取出的消息2----"+msg);
    }
}

默认使用direct队列,如果要使用广播队列:

复制代码
import org.springframework.amqp.core.Binding;
import org.springframework.amqp.core.BindingBuilder;
import org.springframework.amqp.core.FanoutExchange;
import org.springframework.amqp.core.Queue;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class RabbitFanoutConfig {

    //准备两个队列

    @Bean
    protected Queue queue() {

        return new Queue("gavin");
    }


    @Bean
    protected Queue queue2() {

        return new Queue("zzy");
    }

//内置fanout交换器名称amq.fanout

    @Bean
    protected FanoutExchange fanoutExchange() {

        return new FanoutExchange("amq.fanout");
    }

 //将交换器和队列绑定

    @Bean
    protected Binding fanoutBinding(Queue queue, FanoutExchange fanoutExchange) {

        return BindingBuilder.bind(queue).to(fanoutExchange);
    }

    @Bean
    protected Binding fanoutBinding2(Queue queue2, FanoutExchange fanoutExchange) {

        return BindingBuilder.bind(queue2).to(fanoutExchange);
    }
}

一个交换器可以绑定多个队列;

几个重要的注解/Bean

复制代码
import org.springframework.amqp.core.AmqpTemplate;
@Bean
AmqpTemplate amqpTemplate 
//此模板中有发送消息的方法

-------------------------------------
import org.springframework.amqp.core.Queue;
//注册一个队列direct
  @Bean
    protected Queue queue() {

        return new Queue("gavin");
    }

//注册一个广播交换器
 @Bean
    protected FanoutExchange fanoutExchange() {

        return new FanoutExchange("amq.fanout");
    }
//交换器绑定队列
 //将交换器和队列绑定

    @Bean
    protected Binding fanoutBinding(Queue queue, FanoutExchange fanoutExchange) {

        return BindingBuilder.bind(queue).to(fanoutExchange);
    }
//注册一个topic交换器
 @Bean
    protected TopicExchange topicExchange() {

        return new TopicExchange("amq.topic");
    }
//绑定队列
 //将交换器和队列绑定

    @Bean
    protected Binding TopicBind1(Queue queue, TopicExchange topicExchange) {

        return BindingBuilder.bind(queue).to(topicExchange).with("com.gavin.*");//匹配路由规则
    }

消费者:

复制代码
 @RabbitListener(queues = "gavin")
 该方法从对列中消费消息

消息重复消费原因:

消费完毕后本该向broker发送ack,但是由于网路延迟较高过了broker等待的时间,于是broker会把消息再次投递到consumer

解决方案:

数据库---处理消息前,使用消息主键在表中带有约束的字段中insert,插入成功则消费成功,插入失败则已经消费过了,不再进行消费

Map--单机版的使用ConrrentHashMap ->putifAbsent

Redis --分布式锁

保证消息队列的消费顺序

同一个topic,同一个queue,发的时候让一个线程去发,消费的时候让一个线程去消费,如果多线程暂时无法保证消费的有序性

怎么保证消息发送到同一个queue?

RocketMQ 提供了一个MessageQueueSelector 接口,重写接口方法

RocketMQ如何保证消息不丢失

Producer端:

采用send()同步发消息,发送结果是同步感知的;

Broker端:

设置数显策略为同步刷新策略

集群部署,配置主从,高可用模式

Consumer端:

消费正常后再进行手动ACK确认

相关推荐
我命由我1234544 分钟前
Kotlin 数据容器 - List(List 概述、创建 List、List 核心特性、List 元素访问、List 遍历)
java·开发语言·jvm·windows·java-ee·kotlin·list
武子康3 小时前
Java-80 深入浅出 RPC Dubbo 动态服务降级:从雪崩防护到配置中心秒级生效
java·分布式·后端·spring·微服务·rpc·dubbo
YuTaoShao5 小时前
【LeetCode 热题 100】131. 分割回文串——回溯
java·算法·leetcode·深度优先
源码_V_saaskw6 小时前
JAVA图文短视频交友+自营商城系统源码支持小程序+Android+IOS+H5
java·微信小程序·小程序·uni-app·音视频·交友
超浪的晨6 小时前
Java UDP 通信详解:从基础到实战,彻底掌握无连接网络编程
java·开发语言·后端·学习·个人开发
双力臂4047 小时前
Spring Boot 单元测试进阶:JUnit5 + Mock测试与切片测试实战及覆盖率报告生成
java·spring boot·后端·单元测试
Edingbrugh.南空7 小时前
Aerospike与Redis深度对比:从架构到性能的全方位解析
java·开发语言·spring
QQ_4376643148 小时前
C++11 右值引用 Lambda 表达式
java·开发语言·c++
永卿0018 小时前
设计模式-迭代器模式
java·设计模式·迭代器模式
誰能久伴不乏8 小时前
Linux如何执行系统调用及高效执行系统调用:深入浅出的解析
java·服务器·前端