位置: IT常识 - 正文

防止数据重复提交的6种方法(超简单)!(数据库防止重复数据)

编辑:rootadmin
防止数据重复提交的6种方法(超简单)!

推荐整理分享防止数据重复提交的6种方法(超简单)!(数据库防止重复数据),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:防止数据重复excel,防止数据重复提交,防止数据重复输入的正确公式为,避免数据重复,防止数据重复提示的方法,防止数据重复提交,防止数据重复提示的方法,防止数据重复提示的方法,内容如对您有帮助,希望把文章链接给更多的朋友!

有位朋友,某天突然问磊哥:在 Java 中,防止重复提交最简单的方案是什么?

这句话中包含了两个关键信息,第一:防止重复提交;第二:最简单。

于是磊哥问他,是单机环境还是分布式环境?

得到的反馈是单机环境,那就简单了,于是磊哥就开始装*了。

话不多说,我们先来复现这个问题。

模拟用户场景

根据朋友的反馈,大致的场景是这样的,如下图所示:

简化的模拟代码如下(基于 Spring Boot):

import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RestController;@RequestMapping("/user")@RestControllerpublic class UserController {/*** 被重复请求的方法*/@RequestMapping("/add")public String addUser(String id) {// 业务代码...System.out.println("添加用户ID:" + id);return "执行成功!";}}复制代码

于是磊哥就想到:通过前、后端分别拦截的方式来解决数据重复提交的问题。

前端拦截

前端拦截是指通过 HTML 页面来拦截重复请求,比如在用户点击完“提交”按钮后,我们可以把按钮设置为不可用或者隐藏状态。

防止数据重复提交的6种方法(超简单)!(数据库防止重复数据)

执行效果如下图所示:

前端拦截的实现代码:

<html><script>function subCli(){// 按钮设置为不可用document.getElementById("btn_sub").disabled="disabled";document.getElementById("dv1").innerText = "按钮被点击了~";}</script><body style="margin-top: 100px;margin-left: 100px;"><input id="btn_sub" type="button" value=" 提 交 " onclick="subCli()"><div id="dv1" style="margin-top: 80px;"></div></body></html>复制代码

但前端拦截有一个致命的问题,如果是懂行的程序员或非法用户可以直接绕过前端页面,通过模拟请求来重复提交请求,比如充值了 100 元,重复提交了 10 次变成了 1000 元(瞬间发现了一个致富的好办法)。

所以除了前端拦截一部分正常的误操作之外,后端的拦截也是必不可少。

后端拦截

后端拦截的实现思路是在方法执行之前,先判断此业务是否已经执行过,如果执行过则不再执行,否则就正常执行。

我们将请求的业务 ID 存储在内存中,并且通过添加互斥锁来保证多线程下的程序执行安全,大体实现思路如下图所示:

然而,将数据存储在内存中,最简单的方法就是使用 HashMap 存储,或者是使用 Guava Cache 也是同样的效果,但很显然 HashMap 可以更快的实现功能,所以我们先来实现一个 HashMap 的防重(防止重复)版本。

1.基础版——HashMapimport org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RestController;import java.util.HashMap;import java.util.Map;/*** 普通 Map 版本*/@RequestMapping("/user")@RestControllerpublic class UserController3 {// 缓存 ID 集合private Map<String, Integer> reqCache = new HashMap<>();@RequestMapping("/add")public String addUser(String id) {// 非空判断(忽略)...synchronized (this.getClass()) {// 重复请求判断if (reqCache.containsKey(id)) {// 重复请求System.out.println("请勿重复提交!!!" + id);return "执行失败";}// 存储请求 IDreqCache.put(id, 1);}// 业务代码...System.out.println("添加用户ID:" + id);return "执行成功!";}}复制代码

实现效果如下图所示:

存在的问题:此实现方式有一个致命的问题,因为 HashMap 是无限增长的,因此它会占用越来越多的内存,并且随着 HashMap 数量的增加查找的速度也会降低,所以我们需要实现一个可以自动“清除”过期数据的实现方案。

2.优化版——固定大小的数组

此版本解决了 HashMap 无限增长的问题,它使用数组加下标计数器(reqCacheCounter)的方式,实现了固定数组的循环存储。

当数组存储到最后一位时,将数组的存储下标设置 0,再从头开始存储数据,实现代码如下:

import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RestController;import java.util.Arrays;@RequestMapping("/user")@RestControllerpublic class UserController {private static String[] reqCache = new String[100]; // 请求 ID 存储集合private static Integer reqCacheCounter = 0; // 请求计数器(指示 ID 存储的位置)@RequestMapping("/add")public String addUser(String id) {// 非空判断(忽略)...synchronized (this.getClass()) {// 重复请求判断if (Arrays.asList(reqCache).contains(id)) {// 重复请求System.out.println("请勿重复提交!!!" + id);return "执行失败";}// 记录请求 IDif (reqCacheCounter >= reqCache.length) reqCacheCounter = 0; // 重置计数器reqCache[reqCacheCounter] = id; // 将 ID 保存到缓存reqCacheCounter++; // 下标往后移一位}// 业务代码...System.out.println("添加用户ID:" + id);return "执行成功!";}}复制代码3.扩展版——双重检测锁(DCL)

上一种实现方法将判断和添加业务,都放入 synchronized 中进行加锁操作,这样显然性能不是很高,于是我们可以使用单例中著名的 DCL(Double Checked Locking,双重检测锁)来优化代码的执行效率,实现代码如下:

import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RestController;import java.util.Arrays;@RequestMapping("/user")@RestControllerpublic class UserController {private static String[] reqCache = new String[100]; // 请求 ID 存储集合private static Integer reqCacheCounter = 0; // 请求计数器(指示 ID 存储的位置)@RequestMapping("/add")public String addUser(String id) {// 非空判断(忽略)...// 重复请求判断if (Arrays.asList(reqCache).contains(id)) {// 重复请求System.out.println("请勿重复提交!!!" + id);return "执行失败";}synchronized (this.getClass()) {// 双重检查锁(DCL,double checked locking)提高程序的执行效率if (Arrays.asList(reqCache).contains(id)) {// 重复请求System.out.println("请勿重复提交!!!" + id);return "执行失败";}// 记录请求 IDif (reqCacheCounter >= reqCache.length) reqCacheCounter = 0; // 重置计数器reqCache[reqCacheCounter] = id; // 将 ID 保存到缓存reqCacheCounter++; // 下标往后移一位}// 业务代码...System.out.println("添加用户ID:" + id);return "执行成功!";}}复制代码

注意:DCL 适用于重复提交频繁比较高的业务场景,对于相反的业务场景下 DCL 并不适用。

4.完善版——LRUMap

上面的代码基本已经实现了重复数据的拦截,但显然不够简洁和优雅,比如下标计数器的

本文链接地址:https://www.jiuchutong.com/zhishi/297477.html 转载请保留说明!

上一篇:一篇文章让你了解ADAS-HIL测试方案(一篇文章让你了解什么)

下一篇:HTML零基础入门教程完整版(html零基础入门教程)

  • 支付宝年度账单2020怎么查(支付宝年度账单2018)(支付宝年度账单百科)

    支付宝年度账单2020怎么查(支付宝年度账单2018)(支付宝年度账单百科)

  • 电饭煲显示e3是什么原因(电饭煲显示e3是什么意思)

    电饭煲显示e3是什么原因(电饭煲显示e3是什么意思)

  • 小米高精度定位怎么设置(小米高精度定位支持机型有哪些)

    小米高精度定位怎么设置(小米高精度定位支持机型有哪些)

  • 电脑键盘膜有必要贴吗(电脑键盘膜是贴上去的吗?)

    电脑键盘膜有必要贴吗(电脑键盘膜是贴上去的吗?)

  • sarm是什么(sarm是什么车)

    sarm是什么(sarm是什么车)

  • 华为手机在耳机模式怎么调回来(华为手机在耳机模式怎么退出)

    华为手机在耳机模式怎么调回来(华为手机在耳机模式怎么退出)

  • 抖音要不要加入公会

    抖音要不要加入公会

  • 删除好友红包能退回吗(删除好友红包还能领吗)

    删除好友红包能退回吗(删除好友红包还能领吗)

  • word文档里怎么删除空白页(word文档里怎么添加方框)

    word文档里怎么删除空白页(word文档里怎么添加方框)

  • 屏幕有黄斑是什么原因(屏幕出黄斑)

    屏幕有黄斑是什么原因(屏幕出黄斑)

  • 微信群聊群主退出群还在吗(微信群聊群主退群)

    微信群聊群主退出群还在吗(微信群聊群主退群)

  • 淘宝是不是要确认收货才能申请退款(淘宝是不是要确认收货才能评价)

    淘宝是不是要确认收货才能申请退款(淘宝是不是要确认收货才能评价)

  • 腾讯会议会议号怎么弄(腾讯会议会议号是什么意思)

    腾讯会议会议号怎么弄(腾讯会议会议号是什么意思)

  • 知道手机号能搜快手吗(知道手机号能搜到抖音吗)

    知道手机号能搜快手吗(知道手机号能搜到抖音吗)

  • iphone音量变小(iphone音量变小怎么办)

    iphone音量变小(iphone音量变小怎么办)

  • 红米k30和k30pro区别(红米k30和红米k30pro哪个性价比高)

    红米k30和k30pro区别(红米k30和红米k30pro哪个性价比高)

  • 苹果耳机掉了怎么找(苹果耳机掉了怎么查找位置)

    苹果耳机掉了怎么找(苹果耳机掉了怎么查找位置)

  • 苹果x怎么截图(苹果x怎么截图视频教程)

    苹果x怎么截图(苹果x怎么截图视频教程)

  • 小米8se是双卡双待吗(小米8se双4g)

    小米8se是双卡双待吗(小米8se双4g)

  • 智能收款语音播报器怎么设置(智能收款语音播报器怎么连接手机)

    智能收款语音播报器怎么设置(智能收款语音播报器怎么连接手机)

  • 计算机二级word知识点(计算机二级word是哪个版本)

    计算机二级word知识点(计算机二级word是哪个版本)

  • ios13快捷指令怎么用(快捷指令ios13 使用教程)

    ios13快捷指令怎么用(快捷指令ios13 使用教程)

  • vivo双空间设置(vivo手机双空间双系统)

    vivo双空间设置(vivo手机双空间双系统)

  • word文档表格调整增加一行(word文档表格调整行高后,文字不见了)

    word文档表格调整增加一行(word文档表格调整行高后,文字不见了)

  • python运算符的优先级规则(Python运算符的优先级别)

    python运算符的优先级规则(Python运算符的优先级别)

  • phpcms评论功能怎么用(phpcms api)

    phpcms评论功能怎么用(phpcms api)

  • 房产税的计税依据包含增值税吗
  • 收取职工工会会费收据
  • 赠送货物怎么入账
  • 土地使用税怎么终止
  • 贷款利息未取得发票所得税税前扣除哪年开始
  • 购买展示样品怎么入账
  • 对公受托理财赎回分录
  • 跨年的预付账款如何冲销
  • 视同销售可以抵扣销项税吗?
  • 劳务费个人还需要交税吗
  • 普通发票税收
  • 公司买回来做样衣的服装怎么做会计分录?
  • 速动资产不包括存货和预付账款
  • 小规模税局代增值税专用发票怎么交税
  • 商品促销有关问题
  • win8系统怎么关闭防火墙
  • 什么是汇兑损益,汇兑损益产生的原因有哪些?(10分)
  • switcher.exe - switcher是什么进程
  • php表单系统源码
  • PHP:Memcached::quit()的用法_Memcached类
  • dae是什么文件
  • 车保险理赔的钱多久到账
  • 特兰西瓦尼亚的统一
  • 合伙搞养殖
  • 资产减值准则所规范的资产
  • php计算时间
  • 工业企业如何纳税
  • html中的标签有哪些
  • 坏账的确认条件税务会查吗
  • 信息系统服务属于什么类
  • 损益表填制
  • 特种设备年检费怎么开具发票
  • 中国互联网创业成功的年轻人
  • 印花税计入管理费用还是税金
  • 报税是按照实际收入还是开票
  • 公司油票发票信息怎么查
  • 购买办公楼之后怎么入住
  • sql row_number()over()
  • 主要业务活动是什么意思
  • 生育津贴是否需要缴纳增值税
  • 长期待摊费用的最新账务处理
  • 水电费没有票怎么做账
  • 销售退回如何做账务处理
  • 收到赞助费怎么回复客户
  • 公司使用pos机
  • 公司注销后应收账款转给个人
  • 带息票据贴现时,应将其贴现息直接计入
  • 发明专利权限的期限是多少年
  • 融资方式股权融资
  • 车保险说给返钱是真的吗
  • 跨年收入会计分录
  • 客户从公司借钱怎么拒绝
  • 劳务派遣差额征税如何开票
  • 研发投入算到产值里吗
  • 修改mysql数据库字符集
  • mysql的三种查询方式
  • Centos7 下Mysql5.7.19安装教程详解
  • windows取消ctrl+alt+del
  • windrv.exe
  • win7不能复制文件怎么办
  • 苹果mac浏览器
  • 电脑xp自带的杀毒软件在哪里
  • macbookpro鼠标触控板
  • ControlSet001、ControlSet002以及CurrentControlSet之间有什么区别
  • centos怎么安装
  • linux内核配置文件
  • linux系统怎样
  • django orm sqlalchemy
  • jquery 定位
  • 批处理作用
  • android设置hint颜色
  • js右键菜单
  • 浅谈建筑地基基础加固施工技术亲
  • 要使物体从静止状态转变为运动状态需要对这个物体什么
  • javascript初级教程
  • jquery获取当前点击的元素
  • Android boot.img制作
  • 韩国快递关税
  • 同业借款的主要用途
  • 法治建设的基本原则是什么
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设