mysql幻读和不可重复读的区别_不可重复读和幻读的区别

本文探讨了MySQL中幻读和不可重复读的区别。不可重复读主要涉及update和delete操作,可通过锁行避免;而幻读出现在insert场景,需锁表或采用Serializable隔离级别。数据库通常使用MVCC(多版本并发控制)来解决这两种问题,以优化并发性能。乐观锁和悲观锁是两种并发控制策略,乐观锁通过版本控制减少加锁开销,而悲观锁确保数据读写的独占性。
摘要由CSDN通过智能技术生成

当然,   从总的结果来看,   似乎两者都表现为两次读取的结果不一致.

但如果你从控制的角度来看,   两者的区别就比较大

对于前者,   只需要锁住满足条件的记录

对于后者,   要锁住满足条件及其相近的记录

-----------------------------------------------------------

我这么理解是否可以?

避免不可重复读需要锁行就行

避免幻影读则需要锁表

------------------------

####不可重复读和幻读的区别####

很多人容易搞混不可重复读和幻读,确实这两者有些相似。但不可重复读重点在于update和delete,而幻读的重点在于insert。

如果使用锁机制来实现这两种隔离级别,在可重复读中,该sql第一次读取到数据后,就将这些数据加锁,其它事务无法修改这些数据,就可以实现可重复

读了。但这种方法却无法锁住insert的数据,所以当事务A先前读取了数据,或者修改了全部数据,事务B还是可以insert数据提交,这时事务A就会

发现莫名其妙多了一条之前没有的数据,这就是幻读,不能通过行锁来避免。需要Serializable隔离级别

,读用读锁,写用写锁,读锁和写锁互斥,这么做可以有效的避免幻读、不可重复读、脏读等问题,但会极大的降低数据库的并发能力。

所以说不可重复读和幻读最大的区别,就在于如何通过锁机制来解决他们产生的问题。

上文说的,是使用悲观锁机制来处理这两种问题,但是MySQL、ORACLE、PostgreSQL等成熟的数据库,出于性能考虑,都是使用了以乐观锁为理论基础的MVCC(多版本并发控制)来避免这两种问题。

####悲观锁和乐观锁####

悲观锁

正如其名,它指的是对数据被外界(包括本系统当前的其他事务,以及来自外部系统的事务处理)修改持保守态度,因此,在整个数据处理过程中,将数据处

于锁定状态。悲观锁的实现,往往依靠数据库提供的锁机制(也只有数据库层提供的锁机制才能真正保证数据访问的排他性,否则,即使在本系统中实现了加锁机

制,也无法保证外部系统不会修改数据)。

在悲观锁的情况下,为了保证事务的隔离性,就需要一致性锁定读。读取数据时给加锁,其它事务无法修改这些数据。修改删除数据时也要加锁,其它事务无法读取这些数据。

乐观锁

相对悲观锁而言,乐观锁机制采取了更加宽松的加锁机制。悲观锁大多数情况下依靠数据库的锁机制实现,以保证操作最大程度的独占性。但随之而来的就是数据库性能的大量开销,特别是对长事务而言,这样的开销往往无法承受。

而乐观锁机制在一定程度上解决了这个问题。乐观锁,大多是基于数据版本( Version

)记录机制实现。何谓数据版本?即为数据增加一个版本标识,在基于数据库表的版本解决方案中,一般是通过为数据库表增加一个 “version”

字段来实现。读取出数据时,将此版本号一同读出,之后更新时,对此版本号加一。此时,将提交数据的版本数据与数据库表对应记录的当前版本信息进行比对,如

果提交的数据版本号大于数据库表当前版本号,则予以更新,否则认为是过期数据。

要说明的是,MVCC的实现没有固定的规范,每个数据库都会有不同的实现方式,这里讨论的是InnoDB的MVCC。

MySQL中的不可重复读幻读是两种并发控制问题。 不可重复读(Non-repeatable Read)是指在一个事务中,多次读取同一数据,在这个事务还没有结束时,另外一个事务也修改了这个数据,导致多次读取的结果不一致。换句话说,一个事务在执行期间,另一个事务对数据进行了修改,导致当前事务多次读取的结果不同。 幻读(Phantom Read)是指在一个事务中,多次查询同一范围的数据,在这个事务还没有结束时,另外一个事务插入了符合同一范围的数据,导致多次查询的结果不一致。换句话说,一个事务在执行期间,另一个事务对数据进行了插入或删除操作,导致当前事务多次查询的结果不同。 为了解决不可重复读幻读的问题,MySQL提供了不同的隔离级别。隔离级别定义了一个事务对其他事务的可见性和影响范围。MySQL支持四个隔离级别: 1. 读未提交(Read Uncommitted):最低级别,事务可以读取到其他事务未提交的数据,可能出现脏读、不可重复读幻读问题。 2. 读已提交(Read Committed):默认级别,事务只能读取到其他事务已提交的数据,可以避免脏读,但仍可能出现不可重复读幻读问题。 3. 可重复读(Repeatable Read):事务在执行期间,多次读取同一数据时会返回一致的结果,可以避免不可重复读,但仍可能出现幻读问题。 4. 串行化(Serializable):最高级别,事务依次执行,避免了脏读、不可重复读幻读问题,但牺牲了并发性能。 开发者可以根据具体需求选择合适的隔离级别来解决不可重复读幻读问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值