从红包场景谈事务一致性

今天继续谈一下业务层改造相关的一个场景:领取红包。红包,是这几年最火的一类营销手段,从当年微信红包走红后,无论是电商还是互金,都开始加入了“红包”。红包场景本身并不复杂,但由于涉及到了资金的转移,就会引入事务一致性的问题,而且从我了解到的情况看,很多同学在处理这块业务的时候,刚开始都一不小心犯了错,这也是为什么我要选“红包”场景来展开谈的原因。

为了尽可能简化问题,以“小明领取现金红包20元”为例,整个事务包含如下操作:

检查20元红包是否有效

设置红包状态为“已领取”

小明账户余额加上20元

很明显,A,B,C3个操作,要么全部成功,要么全部失败。但在这个场景下,会有什么潜在的问题呢?

一, 1包多领多个线程同时发送领取同一个红包的操作,红包金额重复添加。这个问题常见外部恶意攻击API,由于缺乏必要的数据一致性保护措施,读取脏数据,导致多领。

二,丢失更新,领取红包的同时,同时进行了购买操作,线程1的事务覆盖线程2事务已经提交的数据,造成线程2事务所做操作丢失:

在提出解决方案前,我们将后端系统架构分为两类:

单机版 Standalone,数据库相同,所有的业务操作在一个容器下。

微服务 红包操作,个人账户操作不在同一个容器内进行,分属不同的服务,拥有各自的数据库。

单机版的解决方案

如下图所示,在应用代码中,对红包与账户余额都上锁,确保红包对象同时只会被一个线程所操作。对红包加锁,如果有多余线程想操作红包,一定需要等待线程1被执行结束,这个时候红包的状态已经被更新到数据库中,根本上杜绝一包多领的情况。对账户余额加锁,如果余额账户在扣款的时候,确保领取红包的操作,不会读取脏数据,造成更新丢失。

微服务下的解决方案

微服务的解决方案引入了消息队列来进行处理。如果红包状态正常,并成功将状态至为“已领取”,且消息已经发送成功,用户服务端开始消费这条消息,如果这个时候出现消费失败或者消费超时,利用消息队列进行重试,直到用户端执行成功,考虑到“一包多领”的问题,整个过程中有可能会出现消息重复的问题。所以我们在微服务中,需要做到以下两点:

1. 消费端处理消息的业务逻辑保持幂等性。

2. 保证每条消息都有唯一编号且保证消息处理成功与去重表的日志同时出现。

在使用MQ过程中,需要注意以下几点:

一个应用尽可能用一个 Topic,消息子类型用 tags 来标识。只有发送消息设置了tags,消费方在订阅消息时,才可以利用 tags 在 broker 做消息过滤。

每个消息在业务层面的唯一标识码,要设置到 keys 字段,方便将来定位消息丢失问题。

消息发送成功或者失败,要打印消息日志,务必要打印 sendresult 和 key 字段。

对于消息不可丢失应用,务必要有消息重发机制。

总结

在OLTP系统领域,我们在很多业务场景下都会面临事务一致性方面的需求,一个看起来简单的功能,内部可能需要调用多个“服务”并操作多个数据库来实现,特别是高并发的情况下,更是考验我们架构中细节处理的时候。这里,没有一个标准的万能答案,不过只要我们能遵循基本的设计原则,都可以在场景下找到解决方案。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容