RabbitMQ高级特性 - 事务消息

文章目录

RabbitMQ 事务消息


概述

RabbitMQ 的 AMQP 协议实现了事务机制,允许开发者保证消息的发送和接收时原子性的,也就是说,要么消息全都发送成功,要么全都发送失败(只与发送方有关).

实现原理

AMQP 事务实现类似于传统数据库事务,允许在一个事务中发送多条消息,并在最后提交或回滚.

  • 事务开始:客户端发送 tx.select 方法,RabbitMQ 开启一个新的事务上下文.
  • 事务内的操作:客户端发送消息 basic.publish,RabbitMQ 将这些消息暂存在内存中,并标记为未提交.
  • 事务提交:客户端发送 tx.commit 方法,RabbitMQ 将所有暂存的消息写入队列,并且如果消息标记为持久化,那么就把消息保存到磁盘.
  • 事务回滚:客户端发送 tx.rollback 方法,RabbitMQ 丢弃所有暂存的消息,不会写入队列.

代码实现

不采用事务

a)配置文件

yml 复制代码
spring:
  application:
    name: rabbitmq
  rabbitmq:
    host: env-base
    port: 5672
    username: root
    password: 1111

b)配置常量

kotlin 复制代码
object MQConst {

    // 事务
    const val TRANS_QUEUE = "trans.queue"

}

c)定义队列

kotlin 复制代码
@Configuration
class MQConfig {

    @Bean
    fun transQueue() = Queue(MQConst.TRANS_QUEUE)

}

d)发送接口

两条消息之间故意触发异常.

kotlin 复制代码
@RestController
@RequestMapping("/mq")
class MQApi(
    val rabbitTemplate: RabbitTemplate,
) {

    @RequestMapping("/trans-close")
    fun transClose(): String {
        rabbitTemplate.convertAndSend("", MQConst.TRANS_QUEUE, "trans msg 1")
        //触发异常
        val a = 1 / 0
        rabbitTemplate.convertAndSend("", MQConst.TRANS_QUEUE, "trans msg 1")
        return "ok"
    }

}

e)效果如下:

此时队列中只有一条消息,说明消息不具备事务特性.

采用事务

a)配置两个 Bean

  1. 由于事务对整个 RabbitTemplate 都会有影响,因此这里重新定义一个 Template bean.
  2. 还需要配置一个 RabbitMQ事务管理器.
kotlin 复制代码
@Configuration
class GlobalConfig {

    @Bean("transRabbitTemplate")
    fun transRabbitTemplate(
        connectionFactory: ConnectionFactory
    ): RabbitTemplate {
        val mq = RabbitTemplate(connectionFactory)
        mq.isChannelTransacted = true // 开启事务机制
        return mq
    }

    @Bean
    fun rabbitTransactionManager(
        connectionFactory: ConnectionFactory
    ): RabbitTransactionManager {
        return RabbitTransactionManager(connectionFactory)
    }

}

b)如下步骤:

  • 使用新定义的事务 mq bean.
  • 在方法上加上 @Transaction 注解.
kotlin 复制代码
@RestController
@RequestMapping("/mq")
class MQApi(
    val transRabbitTemplate: RabbitTemplate,
) {

    @Transactional
    @RequestMapping("/trans-open")
    fun transOpen(): String {
        transRabbitTemplate.convertAndSend("", MQConst.TRANS_QUEUE, "trans msg 1")
        //触发异常
        val a = 1 / 0
        transRabbitTemplate.convertAndSend("", MQConst.TRANS_QUEUE, "trans msg 1")
        return "ok"
    }

}

c)效果演示

相关推荐
怪咖码农27 分钟前
Java分布式幂等性怎么设计?
java·分布式·spring cloud
明达技术28 分钟前
分布式 IO 模块:港口控制主柜的智能 “助手”
分布式
是小崔啊30 分钟前
Spring Cloud 07 - 分布式链路追踪APM
分布式·spring·spring cloud·apm·链路追踪
1.01^10003 小时前
[0689].第04节:Kafka与第三方的集成 – Kafka集成SpringBoot
spring boot·分布式·kafka
朱杰jjj4 小时前
分布式之超时和重试
分布式
明达技术5 小时前
分布式 IO:矿山砂石装备高效控制的新引擎
分布式
s:1036 小时前
【MQ】RabbitMQ 高可用延时功能的探究
分布式·rabbitmq
星星点点洲10 小时前
【RabbitMQ的监听器容器Simple和Direct】 实现和场景区别
分布式·rabbitmq
2501_9032386517 小时前
深入解析 Kafka 消费者偏移量管理
分布式·kafka·个人开发
小句17 小时前
Kafka 中基于 Segment 和 Offset 查找消息的过程
数据库·分布式·kafka