简单来说,事务就是一组原子性的sql语句。这个语句要么全部执行要么都不执行。
特性
-
原子性:确保事务的所有操作要么全部完成要么全部不完成,事务的所有操作都视为一个单一不可分割的单元
场景:UPDATE cs_user SET age = 18 , gender = '女' WHERE id = 4。要么全部更新要么更新失败,不会出现age更新成功,
场景:比如规定某个表的字段age大于等于12小于18时,字段type为青少年,而数据库中存在age=16的时候,type='儿童'。会添加失败进行回滚,保存数据的一致性,约束不变
-
持久性:确保一旦事务提交,数据的改变是永久化的,即使发生系统崩溃也不会造成数据的丢失
-
隔离性:保证事务是彼此独立的,并发执行的事务互不干扰。不同的事务隔离级别会影响并发事务的表现。
事务的并发问题及隔离级别
并发问题
脏读(Dirty Read):
-
一个事务读取了另一个事务尚未提交的数据。如果那个事务回滚了,那么第一个事务读取到的数据就是无效的、不一致的。
-
如:事务A读取了事务B更新的数据,然后B回滚操作,那么A读取到的数据是脏数据
不可重复读(Non-repeatable Read):
-
在一个事务中,两次读取同一数据却得到了不同的结果。这是因为在两次读取之间,另一事务修改了该数据并提交了。(侧重于修改)
-
如事务 A 多次读取同一数据,事务 B 在事务A多次读取的过程中,对数据作了更新并提交,导致事务A多次读取同一数据时,结果 不一致。
幻读(Phantom Read):
-
一个事务在读取某范围内的数据时,另一个事务在该范围内插入或删除了数据,导致该事务再次读取时,数据行数发生变化。(侧重于新增或者删除)
-
系统管理员A将数据库中所有学生的成绩从具体分数改为ABCDE等级,但是系统管理员B就在这个时候插入了一条具体分数的记录,当系统管理员A改结束后发现还有一条记录没有改过来,就好像发生了幻觉一样,这就叫幻读。
基于此就需要有不同的事务级别来解决不同的问题,但是事务的隔离级别越高,系统的性能就越低,需要根据具体业务需要来判定
隔离级别
读未提交(Read Uncommitted):
-
允许一个事务读取另一个事务尚未提交的数据。
-
存在的并发问题:脏读、不可重复读、幻读。
-
并发性最高,但数据一致性最差。
读已提交(Read Committed):
-
只允许一个事务读取另一个事务已提交的数据。
-
存在的并发问题:不可重复读、幻读。
-
比未提交读更高的一致性,但仍然存在一定的数据一致性问题。
可重复读(Repeatable Read)- 默认:
-
确保在同一事务中多次读取同一数据时,数据是相同的。
-
存在的并发问题:幻读。
-
MySQL InnoDB存储引擎的默认隔离级别,解决了脏读和不可重复读问题,但幻读仍可能发生。
序列化(Serializable):
-
强制事务按顺序执行,完全避免了脏读、不可重复读和幻读问题。
-
并发性最低,但数据一致性最高。
-
在这种级别下,事务就像串行执行一样,确保完全的隔离性。
事务隔离级别 | 脏读 | 不可重复读 | 幻读 |
---|---|---|---|
读未提交(read-uncommitted) | 是 | 是 | 是 |
读已提交(read-committed) | 否 | 是 | 是 |
可重复读(repeatable-read) | 否 | 否 | 是 |
序列号(serializable) | 否 | 否 | 否 |