最近工作内容需要向一张表里面写入数据,有两个实现方法,每种方法会运行得到一份结果,两个结果的key会有大部分重复,后面跟的value会有不同。 表格中只允许两个结果中其中的一个key存在,二者选其一,只能更新替代。在往表里写的程序是没有错误的,但最终从表里查询的时候,发现很多key有两份数据结果,思考其原因,可能跟事物隔离有关系,这里讲解下事物及事物隔离。
MySQL 事务主要用于处理操作量大,复杂度高的数据。比如说,在人员管理系统中,你删除一个人员,你即需要删除人员的基本资料,也要删除和该人员相关的信息,如信箱,文章等等,这样,这些数据库操作语句就构成一个事务!
一般来说,事务是必须满足4个条件(ACID)::原子性(Atomicity,或称不可分割性)、一致性(Consistency)、隔离性(Isolation,又称独立性)、持久性(Durability)。
事物用来管理更新,删除,插入语句。常用的事物控制语句有:
常用的事物处理语句:
这里注意一点,根据自身项目需要,将AUTOCOMMIT设置为0或1。
回到文章刚开始的问题,在表中出现了一个key具有两种结果,初步估计是事物隔离的问题。上面简单介绍了事物,以及事物隔离的四个类别,这里详细介绍。 当数据库上有多个事务同时执行的时候,就可能出现脏读(dirty read)、不可重复读(non-repeatable read)、幻读(phantom read)的问题,为了解决这些问题,就有了“隔离级别”的概念。
在谈隔离级别之前,首先要知道,隔离得越严实,效率就会越低。因此很多时候,需要在二者之间寻找一个平衡点。SQL标准的事务隔离级别包括:读未提交(read uncommitted)、读提交(read committed)、可重复读(repeatable read)和串行化(serializable ):
因此,查询写错表的事物隔离类型:
mysql> SELECT @@tx_isolation tableName
结果为:
READ-COMMITTED
可以看到,该表的隔离类型为读提交, 即需要提交后其变化才能被另外的事物看见。造成这种情况出现的问题是两个结果都在往表里写入,当其中一个结果中的某个key写入后没有提交时,第二个结果也从其中查询是否可以插入或更新的条件,没有查询到,就直接结果插入到表中了,这样造成了重复性的插入。解决办法就是两个结果按照顺序进行写表,写完第一个,再写第二个,这样就不会出现这种问题了。