1.事务
事务是由一步或几步数据库操作序列组成逻辑执行单元,这系列操作要么全部执行,要么全部放弃执行,是一个整体。
事务是一组不可再分割的操作集合(工作逻辑单元)
1.1 事务的基本操作
- 开始一个事务:
start transaction;
- 创建一个保存点:
savepoint 保存点名;
- 回到保存点(根据具体情况):
rollback to 保存点名;
eg:
mysql> start transaction; --开启事务
Query OK, 0 rows affected (0.00 sec)
mysql> savepoint aa;
--设置保存点aa Query OK, 0 rows affected (0.00 sec)
mysql> insert into account values(1, '张三', 10);
--添加一条记录 Query OK, 1 row affected (0.00 sec)
mysql> savepoint bb; -- 设置保存点bb
Query OK, 0 rows affected (0.00 sec)
mysql> insert into account values(2, '李四', 10000); --再添加一条记录
Query OK, 1 row affected (0.00 sec)
mysql> select * from account; --俩条记录都存在
+----+--------+----------+
| id | name | balance |
+----+--------+----------+
| 1 | 张三 | 10.00 |
| 2 | 李四 | 10000.00 |
+----+--------+----------+
2 rows in set (0.01 sec)
mysql> rollback to bb; -- 回滚到bb状态
Query OK, 0 rows affected (0.01 sec)
mysql> select * from account; -- 第二条记录已经被回滚
+----+--------+---------+
| id | name | balance |
+----+--------+---------+
| 1 | 张三 | 10.00 |
+----+--------+---------+
1 row in set (0.00 sec)
1.2事务操作注意事项
- 如果没有设置保存点,事务也可以回滚,但只能滚到事务的开始,直接使用rollback(前提是事务没有提交)
- 事务一旦被提交(commit),则不可以回退(rollback)
- 回滚时可以选择回滚到哪个提交点
- InnoDB支持事务,MyISAM不支持事务
1.3 事务的ACID特性
- 原子性(Atomicity):事务是应用中最小的执行单位,就如原子是自然界最小的颗粒,具有不可再分的特性;事务是应用中不可再分的最小逻辑执行体。
- 一致性(Consistency):事务的执行结果,必须使数据库从一个一致的状态,变到另一个一致性状态。当数据只包含事务成功提交的结果时,数据库处于一致性状态;如果系统运行发生中断,某个事务尚未完成而被迫中断,而该为完成的事务对数据库所做的修改已被写入数据库,此时的数据库就处于一种不正确(不一致)的状态。因此一致性是通过原子性来保证的。
- 隔离性(Isolation):各个事务的执行互不干扰,任意一个事务的内部操作对其他并发事务都是隔离的。也就是说,并发事务之间不能看到对方的中间状态,并发执行的事务之间不能互相影响。
- 持久性(Durability):持久性是指一个事务一旦被提交,该事务所对数据库所作的更改便持久的保存在数据库之中,并不会被回滚。
1.4事务的隔离性
当MySQL表被多个线程或者客户端开启各自事务操作数据库中的数据时,MySQL提供了一种机制,可以让不同的事务在操作数据时,具有隔离性,从而保证数据的一致性。
1.4.1无隔离性的问题
- 脏读:
是指当一个事务正在访问数据,并且对数据进行了修改,而这种修改还未提交到数据库中,这是,另一个事务也访问这个数据,然后使用了这个数据,这种情况就叫做数据的脏读。 - 不可重复读:
是指在一个事务内,多次读同一数据。在一个事务还没有结束时,另外一个事务也访问了同一数据;在这俩次读取数据之间,若是第二个事务修改了数据,那么第一个事务俩个督导的数据可能是不一样的。所以称为不可重复读。 - 幻读:
是指当事务不是独立执行时发生的一种现象。例如第一个事务对一个表中的数据进行了修改,这种修改涉及到表中的全部数据行;同时,第二个数据也修改这个表中的数据,这种修改是向表中插入了一行新数据。那么,以后就会发生操作第一个事务的用户发现表中还有未修改的数据行,就好像产生了幻觉一样;因此称为数据的幻读。
1.4.2 事务的隔离级别
事务的隔离级别分为:
- 未提交读(read uncommitted):
另一个事务修改了数据,但尚未提交,而本事务中的SELECT会读到这些未被提交的数据(脏读) - 已提交读(read committed):
本事务读取到的是最新的数据(其他事务提交后的)。问题是,在同一个事务里,前后两次相同的SELECT会读到不同的结果(不重复读)。 - 可重复读(repeatable read):
在同一个事务里,SELECT的结果是事务开始时时间点的状态,因此,同样的SELECT操作读到的结果会是一致的。但是,会有幻读现象。幻读的结果其实和不可重复读是一样的表现,差异就在于,不可重复读,主要是针对其他事务进行了编辑(update)和删除(delete)操作。而幻读主要是针对插入(insert)操作。也就是在一个事务生命周期内,会查询到另外一个事务新插入的数据。 - 串行化(serializable)
读操作会隐式获取共享锁,可以保证不同事务间的互斥。