一文简单介绍mysql的事务、锁以及MVCC

 更新时间:2025年12月22日 12:00:37   作者:WKP9418  
在MySQL的InnoDB存储引擎中,锁是用于管理并发访问的一种机制,MVCC的核心思想是通过维护数据的多个版本来实现并发控制,这篇文章主要介绍了mysql事务、锁以及MVCC的相关资料,需要的朋友可以参考下

四种隔离级别

一般默认是RR,即可重复读。如果是互联网公司,为了避免锁等待,一般是RC。

  1. 读未提交(Read Uncommitted)
    会导致脏读。读到了其他事务还没提交的数

  2. 读已提交RC(Read Committed)
    一个事务只能读取到其他事务已经提交的更改。
    会导致不可重读:由于另一个事务提交的影响,导致两次读取的结果不一样。

  3. 可重复读RR(Repeatable Read)
    默认的隔离级别。在这个级别下,保证在同一个事务内,多次读取同一数据的结果是一致的。
    但是会有幻读问题,幻读是一种特殊的不可重复读。为啥说是特殊的不可重复读呢?因为幻读是范围性的不可重复读。事务A执行select …where前后,如果其他事务执行insert语句插入了满足where条件的记录,那么事务A的两次查询结果会不一样。
    需要注意的是,mysql添加了next-key临键锁(行锁+间隙锁)来解决了幻读。
    4.可序列化
    读加共享锁,写加排他锁,读写互斥。使用的悲观锁的理论,实现简单,数据更加安全,但是并发能力非常差。如果你的业务并发的特别少或者没有并发,同时又要求数据及时可靠的话,可以使用这种模式。

MVCC

明确一点,锁、MVCC这些东西都是为了保证ACID的。

锁机制是通过锁定资源(如表、行)来防止并发操作之间的冲突,确保数据一致性。

加锁完全可以解决上面提到的脏读、不可重复读、幻读的问题。就像“可序列化”隔离级别一样,如果什么都加锁,会导致读写时,必然有一方等待,降低了并发能力。所以为了提高性能,基于乐观锁思想,才采用了MVCC来解决问题。所以数据库是乐观锁+悲观锁一起来使用的。
MVCC(多版本并发控制)则是通过维护数据的多个版本,让读操作无需等待写操作完成,从而提高并发性能,同时保证事务隔离性。(会将每一次修改的数据用一个链表链接好,相当于历史记录)。mysql 数据库中,每一条数据都保存一个创建时的版本号和一个删除版本号,MVCC就是根据这两个来发挥作用的。

多版本并发控制的一个简单使用就是,查询时select version,xxx,xx where id = 1读出来,此时version=1。修改数据时版本号+1,update xx ... where id = 1 and version =2,如果返回1说明修改成功,如果返回0,可以提示用户有冲突等。

快照读/当前读

  • 快照读:读取的是历史数据,非最新的数据
  • 当前读 :读取最新数据
  • 快照读:就是select
    select * from table ….;
  • 当前读:特殊的读操作,插入/更新/删除操作,属于当前读,处理的都是当前的数据,需要加锁。
     
    select * from table where ? lock in share mode;
    select * from table where ? for update;
    insert;
    update ;
    delete;

MVCC:解决了读操作时的不可重复读和幻读
临键锁:解决了写操作时的幻读(当前读、更新、删除)。
为啥这么说呢?
看下,在RR隔离级别下,如何解决不可重复读、幻读呢?可重复读其实读到的是历史数据,这叫快照读。只会读取历史数据,不用加锁。其他事务此时insert或update或delete的话,他们的创建版本号或者删除版本号会改变。此时insert的话,假如创建版本号是10。而另一个RR事务版本号是5。MVCC规定,事务版本号是5的话,select的时候,只能查到 创建版本号<=5的数据。所以不会查到创建版本号是10的数据。其实这也解决了幻读。从这里可以看出,MVCC只能解决快照读场景下面的幻读,不能解决当前读场景下的幻读。举个例子:
·SELECT … FOR UPDATE·,这是当前读,直接加锁需要读取最新的数据,那么MVCC就没啥用了。此时再插入一行新记录,那么select也会查出该条记录来。所以该场景下出现了临键锁。

注意临键锁是加在索引上面的,如果没有索引,可能会锁全表。

锁和MVCC对比

总结 

到此这篇关于mysql事务、锁以及MVCC的文章就介绍到这了,更多相关mysql事务、锁及MVCC内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 在Mac系统上配置MySQL以及Squel Pro

    在Mac系统上配置MySQL以及Squel Pro

    给大家讲述一下如何在MAC苹果系统上配置MYSQL数据库以及Squel Pro的方法。
    2017-11-11
  • SQL insert into语句写法讲解

    SQL insert into语句写法讲解

    这篇文章主要介绍了SQL insert into语句写法讲解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-08-08
  • MySQL亿级大表安全添加字段的实战指南

    MySQL亿级大表安全添加字段的实战指南

    在面对亿级数据的 MySQL 表时,直接执行 ALTER TABLE 添加字段的传统方式可能会导致长时间的表锁定,严重影响业务的正常运行,本文将为你提供几种安全、高效添加字段的方案,帮助你在不影响业务的情况下完成大表的字段扩展,需要的朋友可以参考下
    2025-05-05
  • MySQL FLOAT不准问题解析

    MySQL FLOAT不准问题解析

    在数据库处理中,使用FLOAT类型存储浮点数时,由于二进制表示的限制,会导致精度损失,解决方法包括使用DOUBLE或DECIMAL类型来提高精度,或避免使用浮点数进行精确计算,感兴趣的可以了解一下
    2024-09-09
  • MySQL EXPLAIN详细解析

    MySQL EXPLAIN详细解析

    EXPLAIN是SQL性能优化的关键工具,它展示了MySQL如何执行一条SQL 语句,通过分析它的结果,你可以找出查询的瓶颈并进行优化,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧
    2025-11-11
  • 简述MySQL分片中快速数据迁移

    简述MySQL分片中快速数据迁移

    这篇文章主要介绍了MySQL分片中快速数据迁移的相关资料,需要的朋友可以参考下
    2016-03-03
  • 关于MySQL的sql_mode合理设置详解

    关于MySQL的sql_mode合理设置详解

    这篇文章主要给大家介绍了关于MySQL的sql_mode合理设置的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用MySQL具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-07-07
  • 手把手教你用SQL获取年、月、周几、日、时

    手把手教你用SQL获取年、月、周几、日、时

    时间处理是我们日常开发中经常遇到的需求,下面这篇文章主要给大家介绍了关于如何用SQL获取年、月、周几、日、时的相关资料,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2022-12-12
  • mysql8中如何设置sql-mode

    mysql8中如何设置sql-mode

    这篇文章主要介绍了mysql8中如何设置sql-mode问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • MySQL数据库可以用任意ip连接访问的方法

    MySQL数据库可以用任意ip连接访问的方法

    本文主要介绍了MySQL数据库可以用任意ip连接访问的方法,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05

最新评论