李锋镝的博客

  • 首页
  • 时间轴
  • 评论区显眼包🔥
  • 左邻右舍
  • 博友圈
  • 关于我
    • 关于我
    • 另一个网站
    • 我的导航站
    • 网站地图
    • 赞助
  • 留言
  • 🚇开往
Destiny
自是人生长恨水长东
  1. 首页
  2. 原创
  3. 技术
  4. 正文

项目中不用 redis 分布式锁,怎么防止用户重复提交?

2025年9月3日 186点热度 0人点赞 0条评论

1. 前端防抖与按钮禁用(辅助手段)

  • 原理:通过JavaScript限制用户频繁点击,如提交后禁用按钮或添加防抖延迟。
  • 优点:简单易实现,减少无效请求。
  • 缺点:无法防止绕过浏览器的请求(如API工具直接调用)。
  • 示例:
    let isSubmitting = false;
    function submitForm() {
    if (isSubmitting) return;
    isSubmitting = true;
    // 提交逻辑...
    }

2. 令牌机制(Token)

  • 原理:页面加载时生成唯一Token,提交时携带Token,服务端校验后删除或标记为已使用。
  • 适用场景:表单提交、防CSRF攻击。
  • 实现:
    • 生成Token:服务端在渲染页面时生成Token并存储(Session/DB)。
    • 校验Token:提交时验证Token是否存在,存在则处理并删除,否则拒绝。
  • 注意:分布式环境下需共享Token存储(如数据库)。

3. 数据库唯一约束

  • 原理:利用数据库唯一索引防止重复数据插入。
  • 适用场景:创建具有唯一标识的业务数据(如订单号)。
  • 示例:
    CREATE TABLE orders (
    order_id VARCHAR(64) PRIMARY KEY,
    user_id INT,
    ...
    );

    插入前检查order_id 是否存在,或依赖数据库报错处理。

4. 幂等性设计

  • 原理:同一请求多次执行结果一致,需客户端传递唯一业务ID(如订单ID)。
  • 实现:
    • 客户端生成唯一ID(如UUID)并随请求发送。
    • 服务端检查该ID是否已处理:
    • 已处理:返回之前的处理结果。
    • 未处理:执行业务并记录ID。
  • 优点:适用于API接口,天然支持分布式环境。

5. 请求参数哈希去重

  • 原理:对用户ID、操作类型、参数等生成唯一哈希值,短时内拒绝相同哈希请求。
  • 实现:
    • 服务端维护已处理请求的哈希集合(如内存缓存或数据库)。
    • 设置合理的过期时间(如5秒),避免内存泄漏。
  • 示例:
    String hash = MD5(userId + operationType + params);
    if (cache.exists(hash)) throw new RepeatException();
    cache.set(hash, 5); // 5秒过期

6. 乐观锁(针对更新操作)

  • 原理:通过版本号或时间戳控制并发更新。
  • 适用场景:防止数据并发修改(如库存扣减)。
  • 实现:
    UPDATE products 
    SET stock = stock - 1, version = version + 1 
    WHERE id = 100 AND version = 5;

    检查影响行数,若为0则说明数据已变更,拒绝操作。

7. 限流与频率控制

  • 原理:限制同一用户/IP在单位时间内的请求次数。
  • 工具:
    • 单机:Guava RateLimiter。
    • 分布式:数据库滑动窗口计数或令牌桶算法。
  • 示例(滑动窗口):
    SELECT COUNT(*) FROM requests 
    WHERE user_id = 123 AND time > NOW() - INTERVAL 5 SECOND;

8. POST/REDIRECT/GET 模式

  • 原理:提交后重定向到结果页,防止浏览器刷新重复提交。
  • 适用场景:传统Web应用,非API场景。
  • 流程:
    1. 用户提交POST请求。
    2. 服务端处理完成后返回302重定向到结果页(GET请求)。
    3. 用户刷新页面仅重复GET请求,不会重复提交数据。

方案选择建议

  • 简单场景:令牌机制 + 前端防抖。
  • 高并发业务:幂等性设计 + 数据库唯一约束。
  • 更新操作:乐观锁。
  • API接口:强制客户端传递唯一请求ID实现幂等性。
除非注明,否则均为李锋镝的博客原创文章,转载必须以链接形式标明本文链接

本文链接:https://www.lifengdi.com/article/tech/4503

相关文章

  • RedisTemplate和Redisson的区别
  • SpringBoot 实现接口防刷的 5 种实现方案
  • Spring中的Aware接口
  • Redisson分布式锁的watch dog自动续期机制
  • SpringBoot基于redis的分布式锁的实现(源码)
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可
标签: JAVA Redis 分布式锁 后端
最后更新:2025年9月3日

李锋镝

既然选择了远方,便只顾风雨兼程。

打赏 点赞
< 上一篇
下一篇 >

文章评论

1 2 3 4 5 6 7 8 9 11 12 13 14 15 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 46 47 48 49 50 51 52 53 54 55 57 58 60 61 62 63 64 65 66 67 69 72 74 76 77 78 79 80 81 82 85 86 87 90 92 93 94 95 96 97 98 99
取消回复

世间无限丹青手,一片伤心画不成。

那年今日(04月18日)

  • 2007年:前任日本长崎市长伊藤一长逝世
  • 1955年:理论物理学家、相对论创立者爱因斯坦逝世
  • 1955年:周恩来率领中国代表团参加“万隆会议”
  • 1946年:国际联盟决定自行解散,财产移交联合国
  • 1590年:奥斯曼帝国艾哈迈德一世出生
  • 更多历史事件
最新 热点 随机
最新 热点 随机
Everything Claude Code 详细使用文档 配置Jackson使用字段而不是getter/setter来序列化和反序列化 这个域名注册整整十年了,十年时间,真快啊 Claude Code全维度实战指南:从入门到精通,解锁AI编程新范式 Apollo配置中心中的protalDB的作用是什么 org.apache.ibatis.plugin.Interceptor类详细介绍及使用
AI时代,个人技术博客的出路在哪里?使用WireGuard在Ubuntu 24.04系统搭建VPN这个域名注册整整十年了,十年时间,真快啊WordPress实现用户评论等级排行榜插件WordPress网站换了个字体,差点儿把样式换崩了做了一个WordPress文章热力图插件
URL地址末尾加不加“/”有什么区别 突破AI大项目理解瓶颈:三大进阶策略+实战落地指南 微服务架构梳理总结(二) Docker打包部署示例 红黑树简介 为什么MySQL要“小表驱动大表”
标签聚合
设计模式 分布式 数据库 SQL 架构 AI编程 JVM AI Redis Spring MySQL 日常 多线程 WordPress K8s IDEA JAVA ElasticSearch SpringBoot docker
友情链接
  • Blogs·CN
  • Honesty
  • Mr.Sun的博客
  • 临窗旋墨
  • 哥斯拉
  • 彬红茶日记
  • 志文工作室
  • 懋和道人
  • 拾趣博客导航
  • 搬砖日记
  • 旧时繁华
  • 林羽凡
  • 瓦匠个人小站
  • 皮皮社
  • 知向前端
  • 蜗牛工作室
  • 韩小韩博客
  • 风渡言

COPYRIGHT © 2026 lifengdi.com. ALL RIGHTS RESERVED.

域名年龄

Theme Kratos Made By Dylan

津ICP备2024022503号-3

京公网安备11011502039375号