MySQL--事务的四种隔离级别

本文深入解析MySQL事务的四大特性(ACID)及四种隔离级别:Read Uncommitted、Read Committed、Repeatable Read、Serializable,探讨各隔离级别下可能出现的脏读、不可重复读、幻读等问题。

 

MySQL的事务有四种隔离级别:Read Uncommitted,Read Committed,Repeatable Read,Serializable隔离级别越来越严格

基本有begin,commit,rollback这三条命令

怎么设置数据库的事务隔离级别

语句:

SET TRANSACTION ISOLATION LEVEL <READ COMMITTED>;

事务需要隔离是因为事务并发的时候会出现脏读,不可重复读,幻读的问题


事务的四大特性

数据库事务具有ACID这4个特性:

  • A:Atomic,原子性,将所有SQL作为原子工作单元执行,要么全部执行,要么全部不执行;
  • C:Consistent,一致性,事务完成后,所有数据的状态都是一致的,即A账户只要减去了100,B账户则必定加上了100;
  • I:Isolation,隔离性,如果有多个事务并发执行,每个事务作出的修改必须与其他事务隔离;
  • D:Duration,持久性,即事务完成后,对数据库数据的修改被持久化存储。

Read Uncommitted

隔离级别最低的一种事务级别。在这种隔离级别下,一个事务会读到另一个事务更新后但未提交的数据,如果另一个事务回滚,那么当前事务读到的数据就是脏数据,这就是脏读(Dirty Read)

比如事务A进行了更新,但是没有commit,事务B这个时候进行查询,查询的是更新后的数据,然后事务A进行了rollback,事务B在进行查询,查询到的是原始的数据,前后两次查询的数据不一致,这就是脏读


Read Committed

在这种隔离级别下,一个事务可能会遇到不可重复读(Non Repeatable Read)的问题。

不可重复读是指

在一个事务内,多次读同一数据,在这个事务还没有结束时,如果另一个事务恰好修改了这个数据,那么,在第一个事务中,两次读取的数据就可能不一致。

比如事务A,事务B都开始,事务B先进行了一次查询,得到的数据就是原始的数据,然后事务A进行了一次更新,并且提交了,事务B又进行一次查询,得到的数据和第一次又不一样。所以在这种隔离级别下,不可重复读同一条记录


Repeatable Read

在这种隔离级别下,一个事务可能会遇到幻读(Phantom Read)的问题。MySQL默认的事务级别

幻读是指,在一个事务中,第一次查询某条记录,发现没有,但是,当试图更新这条不存在的记录时,竟然能成功,并且,再次读取同一条记录,它就神奇地出现了。

比如一开始数据库中并没有相应的记录,事务B这时候查询这条记录,结果为empty,然后事务A插入了这条记录,并且提交了,然后事务B又开始查询,这时候就可以查询到数据,这就是幻读,而且事务B还可以更新这条记录

至于为什么这种隔离级别就不会出现不可重复读的问题,这是由数据库引擎进行保证的,我们只需要会用的就行了


Serializable

 

这是最严格的隔离级别。在Serializable隔离级别下,所有事务按照次序依次执行,因此,脏读、不可重复读、幻读都不会出现。

虽然Serializable隔离级别下的事务具有最高的安全性,但是,由于事务是串行执行,所以效率会大大下降,应用程序的性能会急剧降低。如果没有特别重要的情景,一般都不会使用Serializable隔离级别。


各种事务隔离级别并发时会出现的问题

另外,有一篇博文讲的很好:链接

在使用数据库开发实际项目时,经常会遇到多个事务并发执行的情况。事务一方面可以保障数据的一致性,但另一方面也可能因为并发带来一些“隐藏的问题”。为了保障数据的正确性,MySQL 提供了不同的事务隔离级别来应对这些问题 [^2]。 事务是为了保证 SQL 之间不产生脏数据。在 innodb 中,默认没有被包裹在事务中的一个单条 SQL 就是一个事务事务具有 ACID 四个特征 [^4]: - 原子性 (Atomicity):一个事务是一个不可分割的工作单元,事务中的所有操作要么全部成功执行,要么全部不执行。如果事务中的任何部分失败,则整个事务将被回滚到事务开始前的状态 [^4]。 - 一致性 (Consistency):在事务开始和结束时,数据库都必须处于一致状态。这意味着事务的执行不会违反任何数据库约束或规则,并且会保持业务逻辑上的一致性。一旦事务完成,它应该使数据库从一个有效状态变为另一个有效状态 [^4]。 - 隔离性 (Isolation):并发执行的多个事务之间互不干扰,如同它们是按顺序独立执行一样。这防止了脏读(读取未提交的数据)、不可重复读(在同一事务内多次读取同一数据时结果不同)和幻读(在事务中多次执行相同的查询时,由于其他已提交事务的插入或删除操作导致结果集发生变化)等现象 [^4]。 - 持久性 (Durability):当事务成功提交后,其对数据库所做的修改将会永久保存,即使系统发生故障(如崩溃、重启等)也不会丢失这些修改。持久性通常是通过日志记录和恢复机制来保证的 [^4]。 事务的隔离性指的是,当并发产生多个事务时,各个事务相互之间相互隔离、互不影响。对于隔离性来说,数据库可以设置不同的隔离级别隔离级别越高,事务之间的相互影响程度就越低,但是并发程度也会随之下降 [^1]。 数据库系统提供了四种事务隔离级别供用户选择 [^5]: - Serializable(串行化):一个事务在执行过程中完全看不到其他事务对数据库所做的更新,事务执行的时候不允许别的事务并发执行,事务串行化执行,只能一个接着一个地执行,而不能并发执行。 - Repeatable Read(可重复读):一个事务在执行过程中可以看到其他事务已经提交的新插入的记录,但是不能看到其他事务对已有记录的更新。 - Read Committed(读已提交数据):一个事务在执行过程中可以看到其他事务已经提交的新插入的记录,而且能看到其他事务已经提交的对已有记录的更新。 - Read Uncommitted(读未提交数据):一个事务在执行过程中可以看到其他事务没有提交的新插入的记录,而且能看到其他事务没有提交的对已有记录的更新。 以下是设置事务隔离级别的示例代码: ```sql -- 请不要在本代码文件中添加空行!!! use testdb1; -- 设置事务隔离级别为 read uncommitted set session transaction isolation level read uncommitted; -- 开启事务 start transaction; insert into dept(name) values('运维部'); -- 回滚事务 rollback; /* 结束 */ ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值