并发环境下mysql插入检查方案

作者:lijiao 时间:2024-01-27 00:59:09 

业务背景:
基本业务场景是这样的,请求数据(车辆vin信息)进入到接口中,需要先判断其在数据库中的状态,如果库中不存在该vin,或者该vin状态位为“1(已完成)”,则执行一些检查操作后,将数据插入到数据库中,此时新增vin状态为0,调用人工处理接口,十分钟后返回结果,将状态置为1。如果其状态位为“0(正在处理)”则驳回操作,返回提示信息。
在单线程环境下,这样的业务没有问题,然而当并发访问接口时,会出现同时进入两条vin相同的请求AB,正常情况应该插入一条A,驳回一条B。然而并发环境下,B执行检查状态时A还没有插入,因此AB都进入到了数据库中,数据就错误了。

解决方案一:
首先想到的是使用sql处理,对数据库对应字段加唯一索引,保证一致性。如果插入重复的数据,则catch该异常,做出提示。


ALTER tableName ADD UNIQUE [indexName] ON (tableColumns(length))

但是由于业务限制,vin在库中是可以重复的,多条重复数据查询最新,所以不能再vin上添加唯一索引。

解决方案二:
使用mysql事务操作,将检查是否存在和插入作为一个事务进行处理,当检查失败的时候,不进行插入。从网上搜索了一下,大致思路如下:


public static void StartTransaction(Connection con, String[] sqls) throws Exception {
   try {
     // 事务开始
     con.setAutoCommit(false);  // 设置连接不自动提交,即用该连接进行的操作都不更新到数据库
     sm = con.createStatement(); // 创建Statement对象

//依次执行传入的SQL语句
     for (int i = 0; i < sqls.length; i++) {
       sm.execute(sqls[i]);// 执行添加事物的语句
     }
     con.commit();  // 提交给数据库处理
     // 事务结束

//捕获执行SQL语句组中的异常  
   } catch (SQLException e) {
     try {
       System.out.println("事务执行失败,进行回滚!\n");
       con.rollback(); // 若前面某条语句出现异常时,进行回滚,取消前面执行的所有操作
     } catch (SQLException e1) {
       e1.printStackTrace();
     }
   } finally {
     sm.close();
   }
 }

但是这样实际上还是没有解决并发的问题,这样只是把两个操作变成了一个原子的sql操作,可以用于同时插入两条数据一致性的情况,但并不适合需求。

既然sql层面没有解决问题,就考虑从java的并发编程方向解决。
解决方案三:
java解决并发问题,首先想到的是使用内置锁或者可重入锁,基本语法如下:
·内置锁:
由于是在Servlet中进行的处理,因此使用synchronized(this)直接处理业务代码,使得并 * 况下,只能有一个线程访问到该段业务代码:


synchronized(this){
 //todo1:检查vin是否存在
 //todo2:如果不存在插入vin
}

·可重入锁:
相当于更灵活的内置锁,在这里与内置锁基本相同


public class DashengCallBack extends HttpServlet {
 private static ReentrantLock lock= new ReentrantLock();
 protected void doGet(HttpServletRequest request, HttpServletResponse response){
   lock.lock();
   try{
     //todo1:检查vin是否存在
     //todo2:如果不存在插入vin
   }finally{
     lock.unlock();
   }
 }
}

经过测试,这个方案是可行的,最终没有采用的原因是因为直接使用这种方式加锁,加锁的代码太多,影响效率。

解决方案四:
设置一个查询Map,插入前存储数据,插入后删除数据,代码如下:


   ConcurrentHashMap<String, String> vinMap=new ConcurrentHashMap<String,String>();
   if(vinMap.containsKey(vin)){
     // todo1: vin 请求完毕后, 从vinInRequestMap里删掉这个vinNo
     // todo2: 返回正在查询
   }
   vinMap.put(vin, "");
   //todo3:插入vin到数据库
   vinMap.remove(vin);
 }

这个方案基本满足了业务需求,唯一的问题是要求接口的更新时间要与业务时间错开,否则更新接口会清空vinMap,导致库中数据混乱,出现错误。

标签:mysql,并发,插入
0
投稿

猜你喜欢

  • 详解MySQL 慢查询

    2024-01-26 19:00:58
  • python使用selenium打开chrome浏览器时带用户登录信息实现过程详解

    2023-07-19 05:00:37
  • 一文带你了解MySQL四大类日志

    2024-01-25 12:53:13
  • python eval 转换k m到乘法计算的操作

    2023-10-30 03:26:04
  • 解决vuex数据异步造成初始化的时候没值报错问题

    2024-04-30 10:46:02
  • python逆向微信指数爬取实现步骤

    2022-03-31 03:36:48
  • Python 读取某个目录下所有的文件实例

    2022-08-27 20:04:13
  • sqlserver数据库导入方法的详细图文教程

    2024-01-15 16:52:24
  • SQL Server 数据库安全管理介绍

    2024-01-19 23:31:01
  • 一文轻松掌握python语言命名规范规则

    2023-03-23 02:11:43
  • 手把手带你用python爬取小姐姐私房照

    2022-04-09 16:02:04
  • 从MySQL数据库表中取出随机数据的代码

    2023-11-14 09:53:58
  • 在MAC OS X上安装MYSQL

    2024-01-28 17:25:46
  • python去除列表中的空值元素实战技巧

    2023-12-08 12:16:06
  • golang 输出重定向:fmt Log,子进程Log,第三方库logrus的详解

    2024-04-27 15:40:14
  • 基于jquery自己写tab滑动门(通用版)

    2024-04-22 22:21:09
  • JS调用打印方法设置页眉页脚的实例

    2024-04-22 12:45:58
  • Mysql中事务ACID的实现原理详解

    2024-01-29 01:49:41
  • Python 排序最长英文单词链(列表中前一个单词末字母是下一个单词的首字母)

    2023-08-26 16:08:33
  • python绘图模块之利用turtle画图

    2022-08-13 01:17:19
  • asp之家 网络编程 m.aspxhome.com