数据库并发问题及四种隔离级别原理深入分析(最详细)

目前网上绝大部分文章只说了并发存在的问题以及隔离级别有哪些,很少有深入分析其具体的实现原理的,本文以最简单实例总结一下。


引言

数据库中事务要遵循的ACID特性:

  • 原子性(Atomicity)
  • 一致性(Consistensy)
  • 隔离性(Isolation)
  • 持久性(Duration)

其中隔离性体现的就是多个事务并发操作数据库时,应该是独立互不干扰的。实际情况中,隔离性是比较灵活的,设定了多个分层级别,主要是为了应对事务并发操作数据库时带来的几个常见问题。


并发常见问题

通常所说的并发问题包括脏读、不可重复读、幻读,这里把脏写也算上,其实和脏读是一类情况。

用两个事务的执行来举例,直观理解。

1 脏读

事务A事务B
UPDATE users SET age = 21 WHERE id = 1;

/* 还未提交 */
SELECT age FROM users WHERE id = 1;

/* age == 21是脏数据 */
ROLLBACK

事务B修改了表中某行的值,但未提交。此时事务A来了读取到了该行被事务B修改后的值,过一会事务B又回滚了。则事务A读取到的值为脏数据,用该数据做的一切操作在回滚后都无效了。

由于读到的数据不是真正写入数据库的,因此称为脏读。

2 脏写

事务A事务B
UPDATE users SET age = 21 WHERE id = 1;

/* 还未提交 */
UPDATE users SET age = 22 WHERE id = 1;

/* 读到脏数据后进行修改*/
ROLLBACK

事务B修改某行数据还未提交时,事务A也修改该行数据。然后事务B回滚了,事务A发现自己的修改无效。

由于写入的这个数据是无效的,因此称为脏写。

3 不可重复读

事务A事务B
SELECT age FROM users WHERE id = 1

/* age == 20是原数据 */
UPDATE users SET age = 21 WHERE id = 1;
COMMIT
SELECT age FROM users WHERE id = 1

/* age == 21是B修改后的数据 */

事务A先读取了表中某行的值,此时事务A还未结束。之后事务B来了修改了该值并且提交了,事务A一会又读了一遍该行值,发现前后不一样了。

由于从数据一致性的角度两次读结果不一样不符合逻辑,因此该问题称为不可重复读

4 幻读

事务A事务B
SELECT * FROM users WHERE age BETWEEN 10 AND 30;

/* A查询年龄在10到30岁之间的人 */
INSERT INTO users VALUES ( 3, ‘Bob’, 27 );

/* B插入了一条数据,年龄27 */
COMMIT
SELECT * FROM users WHERE age BETWEEN 10 AND 30;

/* A再查了一次,发现多了一条数据 */

事务A先查询了一些数据,此时事务A还未结束。事务B来了,插入了一些数据,然后事务A再按之前的条件查询了一次,发现查到的数据变多了。

由于站在事务A的角度看凭空多了一些数据,像幻象一样,因此称为幻读。


四种隔离级别实现原理

为了应对上述并发问题,InnoDB中设置四种隔离级别(Myisam不支持事务):

  • 未提交读——READ UNCOMMITED,解决脏写
  • 已提交读——READ COMMITED,解决脏读
  • 可重复读——REPEATABLE READ,解决脏读、不可重复读
  • 串行化——SERIALIZABLE,解决脏读、不可重复读、幻读

三种隔离级别依次解决每一个并发问题,下面层层递进分析每一种隔离级别的实现原理。

1 未提交读

未提交读是最松的一种并发策略,存在脏读、可重复读、幻读的问题。

如何解决脏写?
某行记录被事务A写过但未提交,则不能让事务B再写

实现措施:

  • 写数据加排他锁,禁止其他事务进行加锁读和加锁写,但允许不加锁读,以此解决脏写
  • 注意,排他锁的作用是防止其他事务再加锁,而不是不允许别人读,如果读数据不加锁,则可以读!!

2 已提交读

已提交读可以解决脏读问题,但还存在可重复读、幻读的问题。

如何解决脏读?
某行记录被事务A写过但未提交,则不能让事务B读到修改后的数据

实现措施:

  • 写数据加排他锁,读数据使用MVCC控制只能读旧版本数据,以此解决脏读
  • 关键点:每次select都获取最新readview

3 可重复读(Mysql默认隔离级别)

可重复读可以解决脏读、不可重复读问题,但还存在幻读的问题。

如何解决不可重复读?
某行记录被事务A写过,则不能让事务读到修改后的数据,不管A是否已提交

实现措施:

  • 写数据加排他锁,读数据使用MVCC控制只能读旧版本数据,以此解决不可重复读
  • 仅在事务第一次select时获取一个readview,一直到事务结束都只使用这一个readview

4 串行化

串行化可以解决脏读、不可重复读、幻读问题,所有问题都解决,但并发度最低

如何解决幻读?
注意,InnoDB的可重复读隔离级别下已经基本解决了幻读问题,通过MVCC+间隙锁的机制。串行化也能解决幻读,但其并发度很低,由于幻读是在其他行插入了新数据,因此直接锁整个表即可。

实现措施:
写数据直接使用表级锁!


总结

关于MVCC具体内容本文不再展开,知道InnoDB的隔离级别就是通过MVCC+锁机制实现的即可!

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
非常感谢您的提问。关于数据库实验数据定义和数据更新实验过程与分析,这个问题比较复杂,需要深入探讨数据库的基础知识和实验操作,因此需要一些文字和图表描述才能更好地解答。对于数据定义方面,需要定义表结构、字段类型和主键等,以及创建表、触发器和存储过程的过程。对于数据更新方面,需要进行增删改查等基本操作,并考虑事务并发控制和索引等技术手段来优化数据库性能。下面我会进一步展开回答您的问题。 数据定义: 数据定义是在数据库系统中建立、修改和删除对象的过程。这些对象可以是表、视图、索引、触发器、存储过程或函数等,我们在定义这些对象的时候需要考虑到它们的结构、约束和功能。举例来说,对于一个表的定义需要包括表的名称、列名、数据类型、主键等,而对于一个存储过程的定义需要包括参数、返回值和逻辑实现等。 在数据定义方面,我们还需要了解和使用 SQL语言。SQL是结构化查询语言(Structured Query Language)的缩写,是操作和处理关系型数据库的标准语言。 对于数据定义,我们可以通过SQL的CREATE语句创建数据库中的表、触发器和存储过程的定义。例如: CREATE TABLE students ( id INT PRIMARY KEY, name VARCHAR(50) NOT NULL, age INT ); CREATE TRIGGER before_insert_students BEFORE INSERT ON students FOR EACH ROW BEGIN IF NEW.age < 0 THEN SET NEW.age = 0; END IF; END; CREATE PROCEDURE get_students_by_age_range( IN min_age INT, IN max_age INT ) BEGIN SELECT * FROM students WHERE age >= min_age AND age <= max_age; END; 这些语句的作用分别是:以 students 表示学生,包含id、name和age等3个字段;在插入students表中的新记录前,检查其age字段的合法性,如果小于0则将其设置为0;查询学生表中符合一定年龄区间查询条件的记录。 数据更新: 数据更新是数据库系统中最常见的操作之一,主要是用来添加、修改和删除数据库中的记录。对于数据更新方面,我们同样需要关注一些基本的概念和技术。例如,事务并发控制和索引等。 事务(Transaction)是指在数据库中一组被提交的操作,可以视为一种逻辑单元。它是保证数据库中数据一致性和完整性的一种机制。数据库中的事务具有ACID的特性。即原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)和持久性(Durability)。 并发控制(Concurrency Control)是数据库系统中的一个重要问题,由于多个用户可能同时访问数据库,如果不加控制,很容易就会发生数据不一致或数据损坏的问题。常见的控制机制有锁(Locking)和事务隔离级别(Transaction Isolation Level)等。 索引(Index)是数据库中常用的优化手段,它可以加速数据的访问和查询。在大型的数据库系统中,索引的效率问题往往是制约数据库性能的瓶颈。 对于数据更新,我们同样可通过SQL语句实现。例如: INSERT INTO students (id, name, age) VALUES (1, 'Tom', 18); DELETE FROM students WHERE id = 1; UPDATE students SET age = 20 WHERE id = 1; 这些语句分别是:向 students 表中插入一条新纪录;从 students 表中删除一个id为1的记录;将 students 表中id为1的记录的age字段更新为20。 在实验数据的分析方面,我们需要利用SQL工具将原始数据导入数据库,并通过SQL语句进行查询和分析。SQL具有强大的查询功能和分析能力,可以帮助我们找出数据的规律和规律关系。例如,在students表中分析出和年龄和成绩之间的相关性,可以使用如下SQL语句: SELECT age, AVG(score) FROM scores GROUP BY age; 这个语句将年龄和成绩进行分组统计,并求出各组年龄对应的平均成绩。 最后需要指出的是,数据库实验的数据定义和数据更新实验过程与分析是数据库知识的基础,需要我们认真学习和实践。如有任何问题或不理解的地方,欢迎在后面提问,我会尽力解答。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值