重复提交导致插入数据库时跳过了验证,如果解决?
点击注册按钮,执行以下代码:
//开启事务
DB::beginTransaction();
try{
if(DB::table('user')->where('email',$email)->exists()){ //如果邮箱已存在,返回错误
DB::rollBack();
return 0;
}
//提交到数据库
DB::table('user')->insert($param);
DB::commit();
return 1;
}catch (\Throwable $e){
DB::rollBack();
return 0;
}
看似很简单的逻辑,但实际使用上,如果点“注册”按钮,点击快了,会出现下图这种情况,重复的邮箱跳过了验证,直接插入到数据库了。
一般这种问题该如何解决?
由于业务需要,还不能给email设置唯一索引,在业务层有方案解决吗
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(5)
唯一索引是最方便的.
实在没办法加唯一索引的话, 可以考虑下面的一个或者多种方案的结合:
点击之后,在收到返回状态之前,弹出模态框并禁止限制重复后台请求呗。
mysql 的话可以用
INSERT INTO table(field1, field2, fieldn) SELECT 'field1', 'field2', 'fieldn' FROM DUAL WHERE NOT EXISTS(SELECT field FROM table WHERE field = ?)
参考 https://my.oschina.net/jsan/blog/270161/
可以针对该接口做幂等性处理,将form params 进行md5, 然后锁3-5秒。就可大量避免该操作。
1.redis做个悲观锁
2.前端做个加载loading 防止重复点击(这个主要辅助作用 实际还需要后端判断)
3.mysql
select for update
做个行锁以上别人的所有的回答 都很详细 可以参照