专栏电商日志财经减肥爱情
投稿投诉
爱情常识
搭配分娩
减肥两性
孕期塑形
财经教案
论文美文
日志体育
养生学堂
电商科学
头戴业界
专栏星座
用品音乐

消息队列之事务消息,RocketMQ和Kafka是如何做的?

  今天我们来谈一谈消息队列的事务消息,一说起事务相信大家都不陌生,脑海里蹦出来的就是 ACID。
  通常我们理解的事务就是为了一些更新操作要么都成功,要么都失败,不会有中间状态的产生,而 ACID 是一个严格的事务实现的定义,不过在单体系统时候一般都不会严格的遵循 ACID 的约束来实现事务,更别说分布式系统了。
  分布式系统往往只能妥协到最终一致性 ,保证数据最终的完整性和一致性,主要原因就是实力不允许...因为可用性为王。
  而且要保证完全版的事务实现代价很大,你想想要维护这么多系统的数据,不允许有中间状态数据可以被读取,所有的操作必须不可分割,这意味着一个事务的执行是阻塞的,资源是被长时间锁定的。
  在高并发情况下资源被长时间的占用,就是致命的伤害,举一个有味道的例子,如厕高峰期,好了懂得都懂。
  对了, ACID 是什么还不太清楚的同学,赶紧去查一查,这里我就不展开说了。  分布式事务
  那说到分布式事务,常见的有 2PC、TCC 和事务消息,这篇文章重点就是事务消息,不过 2PC 和 TCC 我稍微提一下。  2PC
  2PC 就是二阶段提交,分别有协调者和参与者两个角色,二阶段分别是准备阶段和提交阶段。
  准备阶段就是协调者向各参与者发送准备命令,这个阶段参与者除了事务的提交啥都做了,而提交阶段就是协调者看看各个参与者准备阶段都 o 不 ok,如果有 ok 那么就向各个参与者发送提交命令,如果有一个不 ok 那么就发送回滚命令。
  这里的重点就是  2PC 只适用于数据库层面的事务 ,什么意思呢?就是你想在数据库里面写一条数据同时又要上传一张图片,这两个操作 2PC 无法保证两个操作满足事务的约束。
  而且 2PC 是一种 强一致性 的分布式事务,它是同步阻塞 的,即在接收到提交或回滚命令之前,所有参与者都是互相等待,特别是执行完准备阶段的时候,此时的资源都是锁定的状态,假如有一个参与者卡了很久,其他参与者都得等它,产生长时间资源锁定状态下的阻塞 。
  总体而言效率低 ,并且存在单点故障 问题,协调者是就是那个单点,并且在极端条件下存在数据不一致 的风险,例如某个参与者未收到提交命令,此时宕机了,恢复之后数据是回滚的,而其他参与者其实都已经执行了提交事务的命令了。 TCC
  TCC 能保证业务层面的事务 ,也就是说它不仅仅是数据库层面,上面的上传图片这种操作它也能做。
  TCC 分为三个阶段 try - confirm - cancel,简单的说就是每个业务都需要有这三个方法,先都执行 try 方法,这一阶段不会做真正的业务操作,只是先占个坑,什么意思呢?比如打算加 10 个积分,那先在预添加字段加上这 10 积分,这个时候用户账上的积分其实是没有增加的。
  然后如果都 try 成功了那么就执行 confirm 方法,大家都来做真正的业务操作,如果有一个 try 失败了那么大家都执行 cancel 操作,来撤回刚才的修改。
  可以看到  TCC 其实对业务的耦合性很大 ,因为业务上需要做一定的改造才能完成这三个方法,这其实就是 TCC 的缺点,并且 confirm 和 cancel 操作要注意幂等 ,因为到执行这两步的时候没有退路,是务必要完成的,因此需要有重试机制,所以需要保证方法幂等。 事务消息
  事务消息就是今天文章的主角了,它 主要是适用于异步更新的场景,并且对数据实时性要求不高的地方 。
  它的目的是为了 解决消息生产者与消息消费者的数据一致性问题。
  比如你点外卖,我们先选了炸鸡加入购物车,又选了瓶可乐,然后下单,付完款这个流程就结束了。
  而购物车里面的数据就很适合用消息通知异步删除,因为一般而言我们下完单不会再去点开这个店家的菜单,而且就算点开了购物车里还有这些菜品也没有关系,影响不大。
  我们希望的就是下单成功之后购物车的菜品最终会被删除,所以要点就是 下单和发消息这两个步骤要么都成功要么都失败 。 RocketMQ 事务消息
  我们先来看一下 RocketMQ 是如何实现事务消息的。
  RocketMQ 的事务消息也可以被认为是一个两阶段提交,简单的说就是在事务开始的时候会先发送一个半消息给 Broker。
  半消息的意思就是这个消息此时对 Consumer 是不可见的,而且也不是存在真正要发送的队列中,而是一个特殊队列。
  发送完半消息之后再执行本地事务,再根据本地事务的执行结果来决定是向 Broker 发送提交消息,还是发送回滚消息。
  此时有人说这一步发送提交或者回滚消息失败了怎么办?
  影响不大, Broker 会定时的向 Producer 来反查这个事务是否成功 ,具体的就是 Producer 需要暴露一个接口,通过这个接口 Broker 可以得知事务到底有没有执行成功,没成功就返回未知,因为有可能事务还在执行,会进行多次查询。
  如果成功那么就将半消息恢复到正常要发送的队列中,这样消费者就可以消费这条消息了。
  我们再来简单的看下如何使用,我根据官网示例代码简化了下。
  可以看到使用起来还是很简便直观的,无非就是多加个反查事务结果的方法,然后把本地事务执行的过程写在 TransationListener 里面。
  至此 RocketMQ 事务消息大致的流程已经清晰了,我们画一张整体的流程图来过一遍,其实到第四步这个消息要么就是正常的消息,要么就是抛弃什么都不存在,此时这个事务消息已经结束它的生命周期了。
  RocketMQ 事务消息源码分析
  然后我们再从源码的角度来看看到底是怎么做的,首先我们看下 sendMessageInTransaction  方法,方法有点长,不过没有关系结构还是很清晰的。
  流程也就是我们上面分析的,将消息塞入一些属性,标明此时这个消息还是半消息,然后发送至 Broker,然后执行本地事务,然后将本地事务的执行状态发送给 Broker ,我们现在 再来看下 Broker 到底是怎么处理这个消息的 。
  在 Broker 的 SendMessageProcessor#sendMessage 中会处理这个半消息请求,因为今天主要分析的是事务消息,所以其他流程不做分析,我大致的说一下原理。
  简单的说就是 sendMessage 中查到接受来的消息的属性里面 MessageConst.PROPERTY_TRANSACTION_PREPARED  是 true ,那么可以得知这个消息是事务消息,然后再判断一下这条消息是否超过最大消费次数,是否要延迟,Broker 是否接受事务消息等操作后,将这条消息真正的 topic 和队列存入属性中,然后重置消息的 topic 为RMQ_SYS_TRANS_HALF_TOPIC ,并且队列是 0 的队列中,使得消费者无法读取这个消息。
  以上就是整体处理半消息的流程,我们来看一下源码。
  就是来了波狸猫换太子,其实延时消息也是这么实现的,最终将换了皮的消息入盘。
  Broker 处理提交或者回滚消息的处理方法是  EndTransactionProcessor#processRequest ,我们来看一看它做了什么操作。
  可以看到,如果是提交事务就是把皮再换回来写入真正的topic所属的队列中,供消费者消费,如果是回滚则是将半消息记录到一个 half_op 主题下,到时候后台服务扫描半消息的时候就依据其来判断这个消息已经处理过了。
  那个后台服务就是  TransactionalMessageCheckService  服务,它会定时的扫描半消息队列,去请求反查接口看看事务成功了没,具体执行的就是TransactionalMessageServiceImpl#check  方法。
  我大致说一下流程,这一步骤其实涉及到的代码很多,我就不贴代码了,有兴趣的同学自行了解。不过我相信用语言也是能说清楚的。
  首先取半消息 topic 即 RMQ_SYS_TRANS_HALF_TOPIC 下的所有队列,如果还记得上面内容的话,就知道半消息写入的队列是 id 是 0 的这个队列,然后取出这个队列对应的 half_op 主题下的队列,即 RMQ_SYS_TRANS_OP_HALF_TOPIC  主题下的队列。
  这个 half_op 主要是为了记录这个事务消息已经被处理过,也就是说已经得知此事务消息是提交的还是回滚的消息会被记录在 half_op 中。
  然后调用  fillOpRemoveMap  方法,从 half_op 取一批已经处理过的消息来去重,将那些没有记录在 half_op 里面的半消息调用 putBackHalfMsgQueue  又写入了 commitlog 中,然后发送事务反查请求,这个反查请求也是 oneWay,即不会等待响应。当然此时的半消息队列的消费 offset 也会推进。
  然后 producer 中的 ClientRemotingProcessor#processRequest 会处理这个请求,会把任务扔到 TransactionMQProducer 的线程池中进行,最终会调用上面我们发消息时候定义的 checkLocalTransactionState  方法,然后将事务状态发送给 Broker,也是用 oneWay 的方式。
  看到这里相信大家会有一些疑问,比如为什么要有个 half_op ,为什么半消息处理了还要再写入 commitlog 中别急听我一一道来。
  首先  RocketMQ 的设计就是顺序追加写入,所以说不会更改已经入盘的消息 ,那事务消息又需要更新反查的次数,超过一定反查失败就判定事务回滚。
  因此每一次要反查的时候就将以前的半消息再入盘一次,并且往前推进消费进度。而 half_op 又会记录每一次反查的结果,不论是提交还是回滚都会记录,因此下一次还循环到处理此半消息的时候,可以从 half_op 得知此事务已经结束了,因此就被过滤掉不需要处理了。
  如果得到的反查的结果是 UNKNOW,那 half_op 中也不会记录此结果,因此还能再次反查,并且更新反查次数。
  到现在整个流程已经清晰了,我再画个图总结一下 Broker 的事务处理流程。
  Kafka 事务消息
  Kafka 的事务消息和 RocketMQ 的事务消息又不一样了,RocketMQ 解决的是本地事务的执行和发消息这两个动作满足事务的约束。
  而 Kafka 事务消息则是用在一次事务中需要发送多个消息的情况,保证多个消息之间的事务约束,即多条消息要么都发送成功,要么都发送失败,就像下面代码所演示的。
  Kafka 的事务基本上是配合其幂等机制来实现 Exactly Once 语义的 ,所以说 Kafka 的事务消息不是我们想的那种事务消息,RocketMQ 的才是。
  讲到这我就想扯一下了,说到这个 Exactly Once 其实不太清楚的同学很容易会误解。
  我们知道消息可靠性有三种,分别是最多一次、恰好一次、最少一次,之前在消息队列连环问的文章我已经提到了基本上我们都是用最少一次然后配合消费者端的幂等来实现恰好一次。
  消息恰好被消费一次当然我们所有人追求的,但是之前文章我已经从各方面已经分析过了,基本上难以达到。
  而 Kafka 竟说它能实现 Exactly Once?这么牛啤吗?这其实是 Kafka 的一个噱头,你要说他错,他还真没错,你要说他对但是他实现的 Exactly Once 不是你心中想的那个 Exactly Once。
  它的恰好一次只能存在一种场景,就是从  Kafka 作为消息源,然后做了一番操作之后,再写入 Kafka 中 。
  那他是如何实现恰好一次的?就是通过幂等,和我们在业务上实现的一样通过一个唯一 Id, 然后记录下来,如果已经记录过了就不写入,这样来保证恰好一次。
  所以说  Kafka 实现的是在特定场景下的恰好一次,不是我们所想的利用 Kafka 来发送消息,那么这条消息只会恰巧被消费一次 。
  这其实和 Redis 说他实现事务了一样,也不是我们心想的事务。
  所以开源软件说啥啥特性开发出来了,我们一味的相信,因此其往往都是残血的或者在特殊的场景下才能满足,不要被误导了,不能相信表面上的描述,还得详细的看看文档或者源码。
  不过从另一个角度看也无可厚非,作为一个开源软件肯定是想更多的人用,我也没说谎呀,我文档上写的很清楚的,这标题也没骗人吧?
  确实,比如你点进震惊xxxx标题的文章,人家也没骗你啥,他自己确实震惊的呢。
  再回来谈 Kafka 的事务消息,所以说这个事务消息不是我们想要的那个事务消息,其实不是今天的主题了,不过我还是简单的说一下。
  Kafka 的事务有事务协调者角色,事务协调者其实就是 Broker 的一部分。
  在开始事务的时候,生产者会向事务协调者发起请求表示事务开启,事务协调者会将这个消息记录到特殊的日志-事务日志中,然后生产者再发送真正想要发送的消息,这里 Kafka 和 RocketMQ 处理不一样,Kafka 会像对待正常消息一样处理这些事务消息, 由消费端来过滤这个消息 。
  然后发送完毕之后生产者会向事务协调者发送提交或者回滚请求,由事务协调者来进行两阶段提交,如果是提交那么会先执行预提交,即把事务的状态置为预提交然后写入事务日志,然后再向所有事务有关的分区写入一条类似事务结束的消息,这样消费端消费到这个消息的时候就知道事务好了,可以把消息放出来了。
  最后协调者会向事务日志中再记一条事务结束信息,至此 Kafka 事务就完成了,我拿 confluent.io 上的图来总结一下这个流程。
  最后
  至此我们已经知道了 RocketMQ 和 Kakfa 的事务消息全流程,可以看到 RocketMQ 的事务消息才是我们想要的,当然你要是用的流式计算那么 Kakfa 的事务消息也是你想要的。
  需要贴代码的文章其实很难受,这贴的多不好,贴的少又怕不清晰,真的难,如果觉得文章不错记得点个在看哟。
  这篇文章来自于一位技术专家 yes 的公众号 「yes的练级攻略」。
  原文链接:https://mp.weixin.qq.com/s/QMU-ah9f0DjVmLmJiN539Q

每日游讯7月15日农历六月初六星期四工作顺利关注每日游讯,只发布最简短的游戏资讯!本期封面1。死亡细胞移动版于今年2月份在中国国内上线,经过5个月的时间在国内销量突破200万份。2。PlayStation商城美服最后的生还者每日游讯7月29日农历六月二十星期四工作顺心关注每日游讯,只发布最简短的游戏资讯!1。凯娜精神之桥原计划于8月24日登陆PS5和PC平台,开发商通过官方社交媒体发布消息,宣布该游戏将延期到9月21日发售,延期一个月。2。发行steam商店疯狂周三叛乱怪物猎人世界等游戏特惠促销中每日更新steam游戏榜单和新鲜资讯,关注我不迷路目录LEGO建造者之旅叛乱羞辱怪物猎人世界辐射4武装原型LEGO建造者之旅LEGO建造者之旅多半好评支持中文折扣价59LEGO建造冰雪复古传奇类游戏打金需要具备哪些条件?哈喽,大家好,今天来和大家分享一款冰雪复古这款游戏,这个服开了已经有一段时间了,但人气还是很高的,所有的游戏都不会缺少搬砖打金的玩家,传奇类游戏打金的玩家更多,那么要成为专业打金人和越南的比赛,像极了当下中国职业足球的缩影终于能稍微松口气了。在阿联酋沙迦体育场,中国男足凭借伤停补时最后一分钟武磊和王燊超联袂制造的进球,3比2绝杀越南,取得了2022国际足联卡塔尔世界杯亚洲区预选赛最后阶段(以下简称1iFIT推迟IPO募资6。46亿美元计划,欧美股市持续波动或是主因北京时间10月6日,据彭博社报道,健身器材制造商iFITHealthFitness(以下简称iFIT)宣布由于不利的市场条件,将推迟首次公开募股,公司将继续评估拟议发行的时间。8月lululemon10月主题聚焦心理健康,还更加注重社交媒体运营今年年初发布全球幸福感报告后,lululemon在中国市场推出了10月新的主题心悦十月,社交媒体在其中承担了更多的角色。首个活动是一场围绕幸福感话题的直播讨论。10月9日,lulu东鹏特饮跨入咖啡行业,可能对酒饮也有兴趣在能量饮料外,东鹏特饮母公司东鹏饮料集团还在陆续尝试新的产品线,最新的尝试是咖啡。据和讯网10月8日报道,东鹏特饮母公司东鹏饮料集团推出了即饮咖啡品牌大咖。目前,大咖旗下的产品摇摇S11再掀电竞营销热,绝杀越南后武磊又宣新代言DEAL传统体育正处在大赛间的空隙,近期最受关注的赛事非S11莫属,电竞也确实在最近的体育营销中扮演了重要角色,养生品牌日用品甚至漫威都开始试图用电竞吸引新受众。而在消费端,越来越多非体育TopShot开发商DapperLabs与杜兰特合作,未来将发布视频内容和NFT产品北京时间10月13日,NBATopShot开发商DapperLabs与布鲁克林篮网队球员凯文杜兰特(KevinDurant)以及体育媒体公司Boardroom达成了为期两年的协议,银保监会丰富人身保险产品供给,探索开发学生运动意外伤害保险产品北京时间10月15日,中国银行保险监督管理委员会官网发布关于进一步丰富人身保险产品供给的指导意见(以下简称意见)。意见表示将从多领域丰富人身保险产品供给,加大特定人群保障力度。其中
只剩下7好评的三国杀和随处可见的三国杀玩家文丨三明你知道吗,三国杀登陆steam了。那敢情好,它可是不知道多少人的一代回忆啊,现在哪还能找到花几十块就能玩上好几年的良心游戏,肯定很多老玩家跑去steam刷好评吧?不,它已经我的世界玩家打破了速通世界纪录,因为录屏失误被判无效我们都知道我的世界(Minecraft)是由瑞典游戏团队MojangStudios打造的,后来被微软收购了。不过,这并不是坏事,目前这款主打自由创作的沙盒游戏已经在全球创下了超过2哈利波特手游新冬装这就安排!有一种冷,叫玩家觉得女儿冷自从手游哈利波特魔法觉醒实装了冬景,满目银装尽显唯美,不过小伙伴们也表示大冷天的,看着自己的鹅子(女鹅)站在雪地搓手捂脸太心疼,迫切需要冬装救急!巫师小伙子们好歹有件北方来客,实现决战平安京四周年CG都藏了什么大饼?平安京玩家都是列文虎克决战平安京的四周年CG发布了,号称当代列文虎克的平安京玩家坐不住了。短短一支CG暗暗浓缩了多少2022年平安京的活动企划?人有多大胆,地有多大产,能不能从PV中看出来全靠个人悟性,设计师发新图疑似最终幻想16女主角成年版最终幻想系列将在2022年内迎来系列正统续作最终幻想16。虽然这款游戏目前公布的信息并不多,不过玩家们仍然对于制作人吉田直树十分信任,耐心静待佳音。近日,负责为最终幻想系列角色设计焚炙狱神铁甲雄兵张苞全新皮肤今日上线炙焰战将,极寒破军!冷兵器军团竞技策略网游铁甲雄兵张苞狱神系列皮肤今日迎新登场,冰与火两种形态将带来全新视觉体验!除此之外,大世界新玩法首次测试招募火热进行中,即可前往官网参与招募王者荣耀,挑战者杯八强已现,谁能走上15号的决赛舞台挑战者杯淘汰赛分组晋级图经过4天激烈的BO1选拔赛,王者荣耀汇聚了三个赛道(KPLK甲全国大赛)13支队伍的挑战者杯八强终于诞生。这八支队伍将在1月5日1月9日的四天时间内决出两支挑战者杯抽签充满巧合,网友要不是过程透明,就怀疑有剧本了2021年王者荣耀挑战者杯预选赛完美谢幕,经过四日BO1的比赛,TES。AGKXYG和火豹战队成功晋级挑战者杯淘汰赛,与已经确定正赛名额的eStarProTTG狼队和JXG进行挑战能扛能打,又肉又输出,成型即吃分名流卡莎不会运营?赏金猎人不会收菜?新版本上不去分?没关系,今天给大家带来一套能抗能打,有输出又有控制的强力阵容,阵容成型即吃分,重秘发条!这个阵容在现在的这个版本真的非常强力,阵韩媒评选2022最值得关注选手左手李哥登榜竟没有Uzi伴随2021年即将画上尾声,大家目前的眼光都放到了明年2022年的新赛季,S11冬季转会各大赛区都做出了很大的变动,各大赛区大部分队伍也均完成了补强,并以备明年新赛季拿下更加出色的孤影斗神争霸赛惨败!运气极差脸都黑了,吕德华竟然是夺冠热门秦启电竞,畅谈游戏新热点。喜欢看王者荣耀赛事的小伙伴都知道,在刚刚结束的首届KPI杯中,韩涵的刘氏家族大爆冷门击败了吕德华的神鸡营,拿下了最后的总冠军。虽然KPI杯只进行了三个比赛
友情链接:快好知快生活快百科快传网中准网文好找聚热点快软件