我们有个比较古老的 app ,前端是 ember ,后端是 rails
以前一直都好好,最近遇到 2 次,后端收到貌似重复的 api post 请求
除了请求的发送的时间不同,数据传过来是一模一样的
后端有没有办法判断出是 chrome 自动把 api request 重发送了一遍,有这种可能吗
还是说用户又点了一次提交按钮
但照理,提交按钮点一次后就 disable 了啊,实在不懂为什么会出现这种情况
![]() | 1 waterlaw 2022-03-13 00:09:21 +08:00 via Android 有可能卡住了还没 disabled ,用户又点了一次, 我们的做法是 {token}+url 做分布式锁的 key, 锁 1 秒后释放,否则提示请求过于频繁。 |
![]() | 2 waterlaw 2022-03-13 00:13:34 +08:00 via Android 用户输入不可靠,你怎么判断是浏览器问题还是被爬虫了呢? |
![]() | 3 SwinBlackSea 2022-03-13 00:30:37 +08:00 前端可以做个防抖优化下 |
![]() | 4 wellsc 2022-03-13 00:32:05 +08:00 via Android ![]() 保证接口幂等呗 |
![]() | 5 codehz 2022-03-13 00:45:19 +08:00 via Android 加一个 seq 参数,前端 js 确保每次主动调用的时候都是+1 的,后端过滤掉同会话下重复的 seq id(再重复就是业务流程的问题了 |
6 iseki 2022-03-13 02:54:27 +08:00 via Android ![]() api 加个 uuid 和时间戳,比较时间范围和 uuid 重复 |
![]() | 7 elfive 2022-03-13 06:18:52 +08:00 via iPhone ![]() 关键字:幂等性 |
![]() | 8 IvanLi127 2022-03-13 08:20:58 +08:00 via Android 每次请求带一次性 token ,用过后就作废。 |
![]() | 9 xuanbg 2022-03-13 09:20:55 +08:00 保证接口幂等就好了呀,前端是可以做防抖,但那只是对用户体验的优化,不能从根本上解决问题。 |
![]() | 10 ichou 2022-03-13 09:28:13 +08:00 via iPhone 更大的可能性是 js 事件重复绑定,手抖会有时间差的 |
![]() | 11 Chism 2022-03-13 10:15:35 +08:00 via Android 判断 ip ,header ,body 跟上一次一模一样,而且时间间隔很短,例如 5 秒内,都认为是重复请求。 这个可能需要结合 redis 比较方便,ip ,header ,body 内容凑一起 md5 一下作为 key ,要返回的内容作为 value ,重复请求就从 redis 里返回上次的数据。 我没这么做过,纯属想象 |
![]() | 12 ch2 2022-03-13 10:51:47 +08:00 接口在前面做个限流,1 秒 1 次 |
![]() | 13 mostkia 2022-03-13 10:54:27 +08:00 前端回传的表单中,可以额外增加一个计数器参数。每次点击提交按钮进行自增,如果你后台只想要第一次的请求,可以读取这个 index 来进行判断。一般也不用额外操作,保存在某个变量里就行,页面关闭刷新后 rest ,如果需要适应刷新,可以存到 cookie 里面,浏览器关闭后释放 |
![]() | 14 paradoxs 2022-03-13 11:15:32 +08:00 ![]() 直接让前端改就行了啊。 隐藏一个 int i =0; 提交之后+1 , 第二次不让提交或者后端看到 i 不等于 0 就知道怎么处理了啊。 |
![]() | 15 haoliang 2022-03-13 11:37:06 +08:00 如果用户是不是通过正常手段发起的 post 请求,后端防不住啊。 我比较实际,说到 post 幂等,我能想到的具体可操作步骤是:后端响应 html 的 form 中带个 xx 字段,提交时回传,后端插入数据时保证那个 xx 字段唯一。保证字段唯一,可以在数据库表中加字段&unique ,也可以放在 redis 的一个时间桶中,或者其他没想到的方式。 |
![]() | 16 wolfie 2022-03-13 11:38:28 +08:00 根据业务上幂等。 |
![]() | 17 RedBeanIce 2022-03-13 11:39:06 +08:00 根据业务上幂等。 |
![]() | 18 zoyua 2022-03-13 12:01:42 +08:00 via iPhone 前段防抖或者后端针对用户做限流 |
19 Jooooooooo 2022-03-13 12:11:42 +08:00 你把什么叫重复先定义好. |
![]() | 20 pengtdyd 2022-03-13 12:43:26 +08:00 为什么要判断?“除了请求的发送的时间不同,数据传过来是一模一样的” 上面说幂等的我想问一下,难道你们的接口在这种情况下多次请求的结果会不一样??? |
![]() | 21 lower 2022-03-13 12:53:47 +08:00 微信浏览器之前有这种相似的坑,微信浏览器客户端的请求会被微信服务器代理再发到 商户服务器……出现过莫名其妙的重复请求 |
23 Suddoo 2022-03-13 13:03:33 +08:00 不同时间发过来的请求,入库之前先去表里查一下,已存在就直接返回前端,数据已存在 同时发过来的请求,后端加锁,相当于拒绝其中一个请求 |
24 sparky 2022-03-13 19:41:24 +08:00 via Android 用 redis 记录下状态 |
25 a852695 2022-03-13 22:18:01 +08:00 用 seq 来标记,客户端发起协议时候带上 seq ,服务器回复 seq+1 ,客户端更新 seq |
![]() | 26 jdOY 2022-03-13 22:52:30 +08:00 对请求参数做 md5 校验,可以用 redis 控制时效 |
![]() | 27 duduaba 2022-03-14 09:55:13 +08:00 让前端改的都是伪后端,前端改只是用户体验,解决了根本问题了吗? |
![]() | 28 wd 2022-03-14 10:04:08 +08:00 很显然你需要先定义好什么是重复请求,然后我估计解决办法也就有了。 |
![]() | 29 libook 2022-03-14 12:01:56 +08:00 行业共识:前端去重仅用于用户体验优化目的,业务上要想防止多花还是得由后端去重。 |
![]() | 30 Jeyfang 2022-03-14 16:33:11 +08:00 请求头也一样吗 |