脏读、幻读、不可重复读
一、脏读
脏读主要发生在隔离级别很低的事务之间,就比如隔离级别设为Read Uncommitted,也就是读为提交。当我们有多个数据库并发的访问一张表的时候,就会发生脏读。
那么到底什么是脏读?
脏读就是假如,我有两个数据库并发的访问一张表,我的隔离级别是读未提交的,也就是Read Uncommitted,这种隔离级别下,我只要发生数据的更新,另一个并发访问的数据库也就会立马看到结果,这样会发生脏读的情况。就比如,有个学生要转专业,那么我的一个数据库就对这个表的该学生的专业信息进行修改,然后并发访问的另一个数据库就去读数据,发现这个学生的专业已经被修改好了,它就可能下线了,但是由于种种原因,第一个数据库在运行的过程中崩溃了,事务还没有提交,这时候就会发生回滚,该学生的专业的数据又回滚成原专业,但是另一个数据库不知道我发生回滚了,它就认为,这个学生的专业信息已经更改成功了!这样子就产生了脏读的情况,这种影响是十分可怕的!
二、幻读
幻读是也是发生在并发访问同一张表的时候,我第一次查表中的数据,比如我要查公司员工年龄小于30岁的员工有哪些,这时候我的第一个数据库就去查,然后第一次查完之后,并发访问的那个数据库的管理者接到人事通知,有个应届生入职我们公司了,录入一下他的个人信息,然后我就录入了这个员工的个人信息,而这时候,我的第一个数据库觉得,要多查几次,保证我有没有漏掉哪些信息,然后我就去查,发现!怎么又多了一行,我是不是产生幻觉了,刚刚好像没有这一行。
这就是幻读带来的影响,会使得数据库的使用者对数据失去信任感,也造成数据不一致的问题出现。
三、不可重复读
不可重复读通常发生在读提交的隔离级别之下,也就是Read Committed,也就是别人提交了事务,我这边才能看到结果,
就比如:我要查一个员工薪资情况,两个数据库并发访问,第一个数据库对表的数据查询,在第二次查询之前,我的第二个数据库对这名员工的薪资上调,然后提交了事务,第一个数据库现在对表进行第二次查询,发现这个员工的薪资和第一个查到的结果不一样,这就是不可重复读。
四、脏读、幻读、不可重复读的区别
首先这三者都是由于并发访问引起的问题,但是
- 脏读是发生在两个事务之间,由一个事务更改,另一个事务读取,前一个事务发生回滚操作,后一个事务读到的结果不一致的问题
- 幻读是发生在同一个事务内部,通常发生在两次查询操作的中间,另一个数据库对表进行了删除列,插入新的元素,增加列,导致第二次查询的时候,发现和第一次查询结果少了或者多了某些列或者行,就好像产生了幻觉一样。
- 不可重复读也是发生在同一个事务内部,也是发生在二次查询的中间对表进行了操作,他和幻读不一样的点在于,幻读是查的时候多了或者少了某些列,它是发现表中的某些数据和前一次查询的结果不一样的问题。