MySQL中的事务隔离级别详解

 更新时间:2025年05月20日 10:28:59   作者:凭君语未可  
在MySQL中,事务(Transaction)是一个执行单元,它要么完全执行,要么完全回滚,以保证数据的完整性和一致性,下面给大家介绍MySQL中的事务隔离级别详解,感兴趣的朋友一起看看吧

一、事务并发问题

在多个事务同时操作同一份数据时,可能会出现以下几种并发问题:

  • 脏读(Dirty Read):一个事务读取了另一个未提交事务修改的数据,如果后者回滚,则前者读取到的数据是无效的。
  • 不可重复读(Non-repeatable Read):同一个事务中,执行两次相同的查询,由于另一个事务的提交,查询结果不同。
  • 幻读(Phantom Read):一个事务在两次查询之间,另一事务插入或删除了数据,导致前后查询的记录数不一致。

为了避免这些问题,SQL 标准定义了四种事务隔离级别,MySQL 也支持这四种级别。

二、MySQL 事务隔离级别

MySQL 通过 SET TRANSACTION ISOLATION LEVEL 语句来设置事务隔离级别:

SET SESSION TRANSACTION ISOLATION LEVEL <级别>;
SET GLOBAL TRANSACTION ISOLATION LEVEL <级别>;

其中 <级别> 可以是 READ UNCOMMITTEDREAD COMMITTEDREPEATABLE READSERIALIZABLE

1. READ UNCOMMITTED(读未提交)

特点

  • 事务可以读取其他未提交事务的数据。
  • 可能发生“脏读”问题。
  • 性能较好,因为不会使用锁限制读取。

示例

    • 事务A修改了一条记录但尚未提交。
    • 事务B在事务A提交之前读取了修改后的数据。
    • 如果事务A回滚,那么事务B读取到的数据就是“脏数据”。

适用场景

    • 允许读取未提交的数据,适用于不太关心数据一致性的应用,如日志记录、监控数据等。

2. READ COMMITTED(读已提交)

特点

  • 只能读取已经提交的数据,避免了“脏读”。
  • 可能发生“不可重复读”问题。
  • MySQL InnoDB 通过 MVCC(多版本并发控制)来实现此隔离级别。

示例

  • 事务A读取一条记录。
  • 事务B修改该记录并提交。
  • 事务A再次读取该记录,发现数据发生了变化(不可重复读)。

适用场景

    • 适用于大多数 OLTP(在线事务处理)系统,如银行转账、订单管理系统,保证读取的数据是已提交的但允许数据更新。

3. REPEATABLE READ(可重复读)(MySQL 默认级别)

特点

  • 事务内多次读取同一条数据时,数据保持一致(即使其他事务修改并提交了数据)。
  • 通过 MVCC 实现可重复读。
  • 避免了“脏读”和“不可重复读”。
  • 但仍然可能发生“幻读”问题。

示例

  • 事务A第一次读取某条数据。
  • 事务B修改该数据并提交。
  • 事务A再次读取该数据,发现数据未改变(因为事务A读取的是事务开始时的快照)。

如何解决幻读?

  • MySQL InnoDB 通过 间隙锁(Next-Key Locking) 机制来解决幻读问题,即锁住范围,使得其他事务无法插入新的数据,从而防止幻读。

适用场景

    • 适用于高并发场景,尤其是金融行业,如银行账户查询和订单管理系统。

4. SERIALIZABLE(可串行化)

特点

  • 最高级别的隔离,完全避免脏读、不可重复读和幻读。
  • 事务必须依次执行,不能并行,通常会使用 表锁 或 行锁
  • 并发性能极差,适用于对数据一致性要求极高的场景。

示例

  • 事务A读取某一条数据,同时事务B必须等事务A完成后才能读取或修改该数据。

适用场景

    • 适用于需要严格数据一致性的场景,如财务结算、票务系统等。

三、MySQL 默认事务隔离级别

MySQL InnoDB 存储引擎的默认事务隔离级别是 REPEATABLE READ(可重复读),这与 SQL 标准的默认级别(READ COMMITTED)不同。MySQL 通过 MVCC(多版本并发控制)和 间隙锁 解决了幻读问题,因此 REPEATABLE READ 在 MySQL 中比 SQL 标准更强

如果想修改默认的事务隔离级别,可以在 my.cnf(MySQL 配置文件)中修改:

[mysqld]
transaction-isolation = REPEATABLE-READ

或在运行时更改:

SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;

四、不同隔离级别对并发问题的影响

隔离级别脏读(Dirty Read)不可重复读(Non-repeatable Read)幻读(Phantom Read)
READ UNCOMMITTED可能发生 ✅可能发生 ✅可能发生 ✅
READ COMMITTED不会发生 ❌可能发生 ✅可能发生 ✅
REPEATABLE READ不会发生 ❌不会发生 ❌可能发生 ✅(MySQL 中不会)
SERIALIZABLE不会发生 ❌不会发生 ❌不会发生 ❌

五、如何选择合适的隔离级别?

  • READ UNCOMMITTED:适用于对数据一致性要求不高的场景,如日志分析、缓存数据等。
  • READ COMMITTED:适用于大多数业务场景,如电商系统、用户管理系统等,避免脏读,提高并发性能。
  • REPEATABLE READ(MySQL 默认):适用于金融系统、库存管理,保证事务内数据的一致性,防止不可重复读。
  • SERIALIZABLE:适用于对数据一致性要求极高的场景,如银行结算、核心财务系统,但性能损耗较大。

总结

  • READ UNCOMMITTED:可能发生脏读、不可重复读、幻读,性能最高但安全性最低。
  • READ COMMITTED:防止脏读,但可能发生不可重复读和幻读。
  • REPEATABLE READ(MySQL 默认):防止脏读和不可重复读,MySQL 还能避免幻读,适用于大多数高并发业务。
  • SERIALIZABLE:所有并发问题都能避免,但性能最差。

MySQL 默认采用 REPEATABLE READ,主要是因为 MySQL 通过 MVCC 解决了大部分的并发问题,既能保持较高的事务隔离级别,又不会影响太多的性能。

MySQL中的事务隔离级别有四种,分别为:

读未提交(Read Uncommitted)

该级别允许事务读取其他事务尚未提交的数据(脏读)。这意味着一个事务可以读取到另一个事务中间状态的数据,可能会导致数据不一致。

读已提交(Read Committed)

该级别保证事务只能读取到已提交的数据,防止脏读。即使如此,仍然允许发生“不可重复读”(在同一事务中两次读取同一数据,值可能不同,因为另一个事务已经修改了数据并提交)。

可重复读(Repeatable Read)

该级别保证在一个事务中多次读取同一数据时,结果始终一致,避免了“不可重复读”。但是,仍然可能会出现“幻读”(即事务读取的结果集发生了变化,因为另一个事务插入了新的记录)。

串行化(Serializable)

这是最高的隔离级别,强制事务串行执行,即事务排队执行,一个事务在完成之前,其他事务无法访问相同的数据。这可以完全避免脏读、不可重复读和幻读,但性能会较低。

到此这篇关于MySQL中的事务隔离级别有哪些的文章就介绍到这了,更多相关mysql事务隔离级别内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • MySQL 不用存储过程循环插入数据的方法

    MySQL 不用存储过程循环插入数据的方法

    在MySQL中,使用INSERT INTO VALUES语句可以一次性插入多行数据,提高插入效率,还可通过Python的pymysql库生成和执行插入语句,这不仅减少了操作时间,还提高了代码的简洁性和执行效率
    2024-09-09
  • MySQL InnoDB表空间加密示例详解

    MySQL InnoDB表空间加密示例详解

    这篇文章主要给大家介绍了关于MySQL InnoDB表空间加密的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用MySQL具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2020-08-08
  • SQL 语句优化方法30例

    SQL 语句优化方法30例

    在SQL语句优化过程中,我们经常会用到hint,现总结一下在SQL优化过程中常见Oracle HINT的用法.
    2009-10-10
  • MySQL8.0本地服务器连接不上的问题解决

    MySQL8.0本地服务器连接不上的问题解决

    一个内网内连接mysql会出现提示Host xxx is not allowed to connect to this MySQL server,本文主要介绍了MySQL8.0本地服务器连接不上的问题解决,感兴趣的可以了解一下
    2024-01-01
  • 在VB.NET应用中使用MySQL的方法

    在VB.NET应用中使用MySQL的方法

    这篇文章主要介绍了在VB.NET应用中使用MySQL的方法,操作基于Visual Studio IDE进行,需要的朋友可以参考下
    2015-06-06
  • mysql函数全面总结

    mysql函数全面总结

    这篇文章主要介绍了mysql函数,下面文章从MySQL常用的函数开始介绍、还有数值函数,利用举例说明的形式展开内容,需要的朋友可以参考一下
    2021-11-11
  • mysql5.x升级到mysql5.7后导入之前数据库date出错的快速解决方法

    mysql5.x升级到mysql5.7后导入之前数据库date出错的快速解决方法

    这篇文章主要介绍了mysql5.x升级到mysql5.7后导入之前数据库date出错的快速解决方法,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2016-09-09
  • MySQL InnoDB中意向锁的作用及原理

    MySQL InnoDB中意向锁的作用及原理

    意向锁是由InnoDB在操作数据之前自动加的,本文主要介绍了MySQL InnoDB中意向锁的作用及原理,具有一定的参考价值,感兴趣的可以了解一下
    2024-05-05
  • mysql 教程 存储过程

    mysql 教程 存储过程

    最近用mysql + asp.net来写网站,既然mysql已经支持存储过程了,那么像分页这么常用的东西,当然要用存储过程啦
    2009-06-06
  • php连接不上mysql但mysql命令行操作正常的解决方法

    php连接不上mysql但mysql命令行操作正常的解决方法

    这篇文章主要介绍了php连接不上mysql但mysql命令行操作正常的解决方法,需要的朋友可以参考下
    2014-04-04

最新评论