接口幂等设计探索实践

幂等性原本是数学上的概念,即使公式:f(x)=f(f(x)) 能够成立的数学性质。用在编程领域,则意为对同一个系统,使用同样的条件,一次请求和重复的多次请求对系统资源的影响是一致的、或者说是符合预期的。

背景

稳定性设计第一篇:这一小节开始讲设计系统稳定性保证的相关设计,谁都不想自己负责的系统三天两头就出故障,也不想周六日跟女票葡萄美酒夜光杯的时候一个电话call去VPN办公,那么你就想办法让你的系统尽量稳定,我们的

阿里新零售和阿里妈妈,美团,过去我面这些公司都被问过接口幂等相关,接口幂等设计在分布式系统开发中非常常见且很重要,后来我自己做面试官也慢慢意识到幂等的重要性。

一些初学者对幂等这个概念完全不理解,更不知道如何设计,这在工作中很容易给自己惹麻烦,所以一定要会!一定要会!一定要会!

幂等与重复请求区别

幂等:多次请求,在第一次请求不知道结果(比如超时)或者失败的异常情况下,发起多次请求,但却不会因多次请求而出现数据变化。

重复请求:多次请求,第一次请求已经成功的前提下,后续多次进行请求,如果后端服务非幂等服务,则每次请求均会进行数据改变;如果后端是幂等服务,则每次请求返回数据不会变化。

系统里有哪些接口使用到了幂等设计?

问题分析:

幂等的概念首先你肯定理解了,简单通俗易懂,就是无论你是 Http 接口还是 RPC 接口,入参不变的情况下,无论请求多少次,结果都是一样的,请求结果不会因为请求次数不同而改变,没有任务副作用

我:

我参加工作的第一年,在某在线购票(电影票)App的一家公司做后台系统开发,当时我负责积分系统,工作中接到这样一个线上活动需求。业务场景描述:用户每天使用 App 点击签到按钮参加活动,领取相应的积分,每个用户每天只能参加一次签到领积分活动,签到按钮在点击一次后会自设置灰色变为不可点击的状态,这个领积分的接口由我负责开发,提供 API 给 客户端同事,上线后出现这样一个bug,当时没有完善的业务监控系统,功能上线后第二天问出于好奇系统里积分最高的人有多少积分,就在后台跑了一个sql,这一好奇,惊奇的发现有的用户积分高达几万分,因为积分除了签到领取外,大多都是消费累计积分,一块钱才能累积一分,我表示怀疑,什么能人看电影能看几万块钱?

带着这个疑问,我查询了他的积分累积记录,发现大部分积分都是靠签到领积分获得的,按照活动规则,一个人一天只能参加一次签到,不可能有这么多积分,而这个用户一天签到几百次,后来经过和前端一同检查bug发现问题所在,原因是签到按钮虽然变灰,但是请求的 url 没有在前端页面隐藏,用户通过技术手段绕过 button 变灰的前端限制重复刷新了接口,重复获得积分。

事后问题分析:

这个bug最大问题还在我这里,因为我的接口没有做幂等设计,正确的逻辑应该是根据系统当前日期做幂等,幂等后无论用户发起多少次请求,最后的结果都是一样的,积分只累加一次。好在这个bug没有被黑产发现,只有几个用户发现损失可控。

因为我缺少设计经验,不懂幂等设计,领导也没提醒我,所以出现这种bug,经历更多和钱相关的系统开发后,我明白一个道理,任何系统设计,都要考虑业务的安全性,内部系统可以为了节省人力,适当简化设计,做到防君子不防小人,假设你的同事都是君子,对C端用户的系统,不光要防君子,还要防小人,风险防范不能全指望风控系统,有时bug可能会来自系统内部,比如用户并没有恶意盗刷之意,只是网络不好,用户等了两秒钟还没加载完就多点了几次签到按钮,我的接口没有做幂等设计,只要收到请求就会多给用户加积分,这个时候能怪用户吗?很显然是开发者的责任。

关于这个接口的幂等设计,我是这样解决的:

1.积分接口后台根据用户手机号 + userId + 系统当前日期拼接后生成唯一流水号,根据流水号后保存,如果用户重复发起请求,先根据唯一流水号校验在后台做校验,如果流水号存在直接返回上一次请求结果,考虑到并发的情况下,状态判断使用了锁处理。2.开发业务监控系统,采用定时任务每天生成系统里 Top100 积分增长最多名单,运营 or 技术人员每天观察有没有异常。

经过这次bug反思,学习到亮点:

1.理解幂等设计的重要性,凡事和钱相关的功能请谨慎。2.监控系统的重要性,这里的监控说的是业务类监控,如果那天我没有好奇系统里谁的积分最高,这个bug会什么时候发现?

面试官:嚯,有点意思,你还真的是写了个大bug,弄懂了吸取教训就好,可别进了我的项目组后拿我们的系统写这bug。

深入分析:

在编程中一个幂等操作的特点是其任意多次执行所产生的影响均与一次执行的影响相同。幂等函数,或幂等方法,是指可以使用相同参数重复执行,并能获得相同结果的函数。这些函数不会影响系统状态,也不用担心重复执行会对系统造成改变。例如,“setTrue()”函数就是一个幂等函数,无论多次执行,其结果都是一样的.更复杂的操作幂等保证是利用唯一交易号(流水号)实现。

—— 百度百科

如果你了解 Restful 风格接口,相信你对 GET / POST / DELETE 几个动词不陌生,小编在一次面试中,我记得是锤子科技,面试官问我是否了解 Rest 接口,我balabala回答了这几常用的动词,面试官又问我:那你除了知道 GET 是从服务器获取资源,还有别的理解吗?当时我没搭上来,出了公司以后才想起,GET 动作的设计应该是幂等的。同理 DELETE 也是幂等的,如果你设计的接口 GET / DELETE 不是幂等的,那么你可能要重新思考一下了。

1.工作中常见的幂等设计场景

如果你做的功能和钱相关,或者能还钱的,那么你就要小心了,每一个接口都要先考虑下是否需要幂等设计,下面是两种常见的需求场景。

1.发券/积分接口,通常通过 orderId userId 做幂等校验。2.支付/退款接口,我们不希望用户发起多次支付都收到用户的钱,用户会投诉,还要把钱退还给用户,对系统还是客服人员来说都是无用功,支付系统非常复杂,想做好支付系统,还有很多东西需要学习,要考虑网络延迟,服务异常,订单中心回掉超时等各种不稳定的因素,通常采用前端控制,逻辑层状态的控制,数据层唯一索引的控制,以及分布式锁的控制,在幂等篇不过过多讨论。

2.幂等接口常见设计方案

1.客户端按钮提交限制,每次提交一个请求时,按钮置为不可用。2.后台系统逻辑层处理,生成保存唯一ID(流水号),每次请求先校验流水号是否已经存在,存在则表示重复操作,直接返回上一次操作结果。3.token校验机制,客户端请求前先申请token,同一个token只处理一次,无token或者相同token不做处理4.分布式锁,如引入 Redis 分布式锁,防止其他请求重复操作。5.请求队列,引入 MQ 排队的方式让请求有序处理,关于异步操作的应用会在后面的章节讲解。

每一种方案都有自己的优缺点,比如客服端按钮提交限制,实现简单,但是不能同根本上解决问题,后台生成唯一ID,判断存在状态必须要保证原子操作,可以采用多种方案组合的方式解决幂等问题,我们的目标是,用最容易维护的方法解决问题。

总结

在过去的工作经历中,我招进来一个工作三年的同事,场景是开发一个退款接口,review代码的时候,我发现退款的功能是做完了,钱确实能退,但是并没有做幂等设计,我俩讨论了下,我说:如果同一个订单被请求了两次退款,那这钱是不是要退两次,这很危险呀?当时这个同事并没有意识到这一点,因为没有相关经验,连概念都不知道,作为一个三年经验的实在不应该,和钱相关的功能一定要慎重,做幂等设计就是为了系统能防君子,也要防小人。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/306867.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Windows 7 安装 .NET 5 / .NET Core 3.1 环境的方法和依赖文件

随着 .NET 5 正式版的发布,越来越多的人开始向 .NET 5 、.NET Core 平台迁移。尽管微软已经在 2020 年 1 月 14 日停止了对 Windows 7 的支持,但仍有大批客户的操作系统对 Windows 7 恋恋不舍。为此,我们需要解决 .NET 5 运行时(R…

easyui 动态设置单元格控件_动态显示最大最小值的折线图

小伙伴们好啊,今天和大家分享一个图表有关的知识。折线图想必大家已经司空见惯,今天就要在简单的折线上,做出不简单的效果。用动态折线图,展示一周销售的变化,并且自动突出最大最小值。1、准备数据源以某家水果店1月份…

WebBenchmark动态测试Webapi

在编写Webapi测试用例的时候都是定义固定的测试数据,但这样的测试只能针对单一数据。为了更好的模拟实际情况,往往需要进行动态数据测试;通过动态数据测试可以更好的测出服务在不同数据情况下的处理能力。WebBenchmark支持动态数据函数&#…

linux开发需要学习什么,linux开发需要掌握哪些知识?

原标题:linux开发需要掌握哪些知识?嵌入式linux开发的应用是非常广泛的。而且linux是开源的,各种技术也是非常成熟的。不过很多初学者在学习linux开发过程中感觉非常难,那么对于linux开发需要掌握哪些知识呢?1.Linux是…

csv 字符串_python3从零学习-5.5.1、CSV 文件读写

源代码: Lib/csv.py模块内容csv 模块定义了以下函数:csv.reader(csvfile, dialectexcel, **fmtparams)返回一个 reader 对象,该对象将逐行遍历 csvfile。csvfile 可以是任何对象,只要这个对象支持 iterator 协议并在每次调用 __ne…

持续交付三:动手自动化“开发”—“测试”

前两篇博文中提到Development,QA,Staging,Production四个环境,也说明了源代码的分支和四个环境的对应关系,本篇博文聊一下,怎么把源码自动化发布到对应的环境中。市面上主流的DevOpt工具都支持这些功能,github,gitlab,…

使用WebBenchmark对webapi进行管理和性能测试

WebBenchmark是基于beetlex开发的webapi管理和性能测试软件,最新版本1.0.3可以独立运行在linux/windows中,并不再需要安装.net core运行环境。部署可以通过以下地址下载最新版本:https://github.com/IKende/WebBenchmark可以根据需要下载linux64或win64运…

三维叉乘怎么算_奇技淫巧系列:向量叉乘

​一般我们在解决立体几何题目时会选择建立坐标系,因为这样做比较保险也有固定套路。很多时候这些题目要求你计算某一个面的法向量(normal vector),这在高中阶段也是有固定方法的,我们这里想要介绍的是一种更高级也更迅…

g++ linux intel 汇编,g++ linux

目标:运行C代码example:有func.h,func.cpp, main.cpp- .h无需编译,但.h中函数实现的地方需要编译(func.cpp)- 逻辑:cpp各自生成可执行文件(.o),再进行链接g -c func.cppg -c main.cppg main.o func.o -o test或者直接:…

听说容器正在吃掉整个软件世界?

过去几年,以 docker、kubernetes 为代表的容器技术已发展为一项通用技术,BAT、滴滴、京东、头条等大厂,都争相把容器和 k8s 项目作为技术重心,试图“放长线钓大鱼”。就说腾讯吧,目前基本所有业务都跑在云上&#xff0…

linux 短信功能,Android调用系统短信功能发送短信

Android调用系统短信功能发送短信有两种方法:第一种,设定发送的号码,和内容,界面没有联系人,群组组等按钮,如下图所示:代码如下:Uri smsToUri Uri.parse("smsto:114");// 联系人地址…

bcm943602cs蓝牙用不了_原来手机的蓝牙功能这么强大!除了连接耳机,还有这六大实用功能...

蓝牙是手机上的一个普通功能,基本上所有的手机里都有它,原先它的作用很有限,只是用来传输数据,但由于速度太慢,最后也是被软件所淘汰,慢慢的可以用来连接耳机,这也是博主用的比较多的一个功能&a…

数据结构——表达式求值(中序)

表达式求值(中序) 实验二 基于栈的中缀算术表达式求值 【实验目的】 1.掌握栈的基本操作算法的实现,包括栈初始化、进栈、出栈、取栈顶元素等。 2.掌握利用栈实现中缀表达式求值的算法。 【实验内容】 问题描述 输入一个中缀算术表达式,求解表达式的值。…

msdn画圆弧函数_画直线不简单!python-matplotlib告诉你为什么

1 说明:1.1 python的matplotlib画直线,看似简单,其实很难,从简单到复杂,逐步深入,小白秒懂。1.2 内容:画直线,画圆,画圆点,动画的单摆和圆套圆,好…

Wifi6网络

2020年是Wifi6设备全面爆发的一年,华为、小米、华硕、腾达、TP-LINK、360等多家厂商相继发布了Wifi6路由产品,掀起了一股更换路由器的热潮。首先,我们先来看几个常识1、Wifi6和IPv6两个没有必然联系,Wifi6是一种支持802.11ax的Wif…

数据结构——用栈解决回文字符问题

回文 回文是指正读反读均相同的字符序列,如“abba”和“abdba”均是回文,但“good”不是回文。试写一个算法判定给定的字符序列是否为回文。(提示:将一半字符入栈。) 所需的知识前提:栈 以下是顺序栈的基本算法 结构…

aspose excel中文文档_除了VBA,还有哪些编程语言可以操作Excel文件?

Excel(Microsoft office)是现在最常用的办公软件,主要涉及电子表格制作、数据处理、报表输出展示以及更高端的还有金融建模等;我们知道,在需要批处理多个Excel工作表以及工作簿的时候,需要用到一个自动化的利器:VBAVBA…

关于.NET5在IIS中部署的几个问题总结

本来我的系列教程已经慢慢剥离开IIS了,毕竟有了Docker容器以后,配合Nginx使用真的很不错。但是还是有很多同学使用IIS的,这个不可否认IIS的重要性。随着.NET的发布,很多小伙伴已经开始升级了,我也就陆陆续续收到了一些…

数据结构——括号匹配问题

括号匹配 给定一个字符串,其中的字符只包含三种括号:花括号{ }、中括号[ ]、圆括号( ),即它仅由 “( ) [ ] { }” 这六个字符组成。设计算法,判断该字符串是否有效,即字符串中括号是否匹配。括号匹配要求括号必须以正…

wordpress多站点主站调用分站最新文章_企业网站SEO最新的7个优化步骤!

如果你是一个企业主,你有建立企业官方网站的经验,在2-3年的运作中,我相信你至少修改了一个网站,甚至做了一个重大的SEO策略调整。当我们开始建立一个公司的时候,很多时候就是认为只要我们有一个公司的网站,…