位置: 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零基础入门教程)

  • vivo手机怎么设置视频通话美颜(vivo手机怎么设置老年模式)

    vivo手机怎么设置视频通话美颜(vivo手机怎么设置老年模式)

  • 华为荣耀20Pro相机怎么自定义水印(华为荣耀20pro相机参数)

    华为荣耀20Pro相机怎么自定义水印(华为荣耀20pro相机参数)

  • iphonexr信息发送失败(苹果xr短信发送失败)

    iphonexr信息发送失败(苹果xr短信发送失败)

  • 白色硅胶手机壳脏了怎么清洗(白色硅胶手机壳发黄怎么清洗变白)

    白色硅胶手机壳脏了怎么清洗(白色硅胶手机壳发黄怎么清洗变白)

  • 华为nova7se防水等级(华为nova7se防水性)

    华为nova7se防水等级(华为nova7se防水性)

  • 苹果蓝牙耳机一边总是没电(苹果蓝牙耳机一般能用几年)

    苹果蓝牙耳机一边总是没电(苹果蓝牙耳机一般能用几年)

  • 抖音dou+是什么(抖音dou+是什么套路)

    抖音dou+是什么(抖音dou+是什么套路)

  • 苹果7plus微信没有提示音(苹果7plus微信没声音怎么调)

    苹果7plus微信没有提示音(苹果7plus微信没声音怎么调)

  • 微信不点开看不到信息怎么回事(微信不点开看不到信息提示)

    微信不点开看不到信息怎么回事(微信不点开看不到信息提示)

  • 拼多多打款补偿怎样领(拼多多打款补偿预扣税20%)

    拼多多打款补偿怎样领(拼多多打款补偿预扣税20%)

  • iphone怎么去掉dock栏(iphone怎么去掉震动)

    iphone怎么去掉dock栏(iphone怎么去掉震动)

  • oppo手机怎么添加插件(oppo手机怎么添加桌面)

    oppo手机怎么添加插件(oppo手机怎么添加桌面)

  • 京东手机下单怎么备注(京东手机下单怎么退款)

    京东手机下单怎么备注(京东手机下单怎么退款)

  • ppt没有保存关闭找回(ppt没有保存关闭了几天后怎么找回来)

    ppt没有保存关闭找回(ppt没有保存关闭了几天后怎么找回来)

  • 一加7oneplus有耳机孔吗(一加7耳机怎么解决)

    一加7oneplus有耳机孔吗(一加7耳机怎么解决)

  • 苹果xsmax美版支持双卡双待吗(苹果xsmax美版支持nfc吗)

    苹果xsmax美版支持双卡双待吗(苹果xsmax美版支持nfc吗)

  • 华为mate30与p30对比(华为mate30跟p30)

    华为mate30与p30对比(华为mate30跟p30)

  • linux时间同步命令

    linux时间同步命令

  • 腾达路由器无线桥接怎么设置(腾达路由器无线中继)

    腾达路由器无线桥接怎么设置(腾达路由器无线中继)

  • 新ipad怎么验机(ipad怎么验机手机是否原装)

    新ipad怎么验机(ipad怎么验机手机是否原装)

  • 描述文件为什么一打开是空白(描述文件为什么要输入密码)

    描述文件为什么一打开是空白(描述文件为什么要输入密码)

  • 前程无忧如何投简历(前程无忧如何投诉招聘平台虚假招聘)

    前程无忧如何投简历(前程无忧如何投诉招聘平台虚假招聘)

  • 华为p10安全模式怎么解除(华为p10plus安全模式)

    华为p10安全模式怎么解除(华为p10plus安全模式)

  • 关于Mac OS X5个屏幕截图功能使用介绍(imac 5k屏幕参数)

    关于Mac OS X5个屏幕截图功能使用介绍(imac 5k屏幕参数)

  • python+cuda编程(一)(python调用cuda执行加法)

    python+cuda编程(一)(python调用cuda执行加法)

  • 帝国cms怎么加自定义页面(帝国cms怎么加自己的名字)

    帝国cms怎么加自定义页面(帝国cms怎么加自己的名字)

  • 个税申报的收入扣除社保吗
  • 公司购入的财务怎么做账
  • 税法规定的增值税
  • 对公账户要收手续费吗
  • 应补退税额是什么意思
  • 独立核算的分公司
  • 筹建期印花税退税分录
  • 管理费用贷方有哪些递减
  • 薪金性支出包括什么
  • 资本公积与什么有关
  • 对外投资企业利润回国意愿的政策建议
  • 包装报废会计处理
  • 借差旅费退回现金怎么办
  • 投资者撤回投资额300000
  • 小规模纳税人自开专票
  • 保理利息开具增值税专用发票
  • 无形资产软件摊销年限的最新规定2021
  • 固定资产上的配件经常更换
  • 分包给总包开什么发票
  • 文化公司发展战略
  • 车辆保险车船税怎么计算
  • 按简易办法征收增值税
  • 怎么证明公司的存在
  • 银行手续费发票税率
  • cookie 区别
  • 右击任务栏启动怎么关闭
  • 只有收据没有发票可以保修吗
  • win10记事本怎么改字体颜色
  • php实现上传图片功能
  • uni-app介绍
  • 专票认证后可以不抵扣吗
  • Yii2中hasOne、hasMany及多对多关联查询的用法详解
  • 共轭梯度法matlab代码
  • 企业和银行未达账项
  • spring boot 2.3.0
  • ie浏览器网页滚动条怎么隐藏
  • SpringBoot+Vue实现在线商城系统
  • 探望生病职工的话语
  • 锅炉维修项目
  • 发票抵扣联和发票联区别
  • 企业注销后还有风险吗
  • 借款利息收入发票
  • 销售部门差旅费报销会计分录
  • 汽车保险费计入应付账款还是其他应付款
  • 销售旧货的增值税是销项税吗
  • 计提缴纳企业所的会计分录怎么写
  • 初级职称需要提交什么资料
  • 红冲的发票能不能取消
  • 建筑附加税税率
  • 教育培训行业税目分类
  • 报销粘贴单怎么粘贴发票
  • 来料加工账务处理流程
  • 建筑企业缴纳资源税
  • 查补以前年度税款账务处理
  • 含有增值税的工作有哪些
  • 增值税申报表里期初未缴税额是什么意思
  • 国库集中支付是什么意思
  • 一般纳税人资格证书怎么查询
  • 政府补助计入资本公积吗
  • 应付账款转入营业外收入的证明
  • 所得税会计凭证包括哪些
  • 现金日记账本月合计怎么划线
  • 劳动关系从什么时候确立
  • 什么是结账?结账前要做哪些准备工作?
  • linux route -n命令结果详解
  • 高县土火锅历史介绍
  • win8计算机管理员权限
  • c:windowssys:em32mtgyu.dll 内存分配访问无效的解决办法
  • linux系统怎么隐藏文件
  • unity3d协程的作用
  • html+css网站设计基础教程
  • python引用方法
  • JS中call/apply、arguments、undefined/null方法详解
  • python语言基本语法
  • 基于javascript的毕业设计选题
  • jquery+ajax实现直接提交表单实例分析
  • 土地增值税税率表
  • 个人纳税记录为多少
  • 多张发票怎么粘贴在a4纸上面
  • 广东佛山税务局人工电话是多少
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

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

    友情链接: 武汉网站建设