在关系型数据库中,事务扮演着很重要的角色。
概念:事务是并发控制的基本单位,其中的操作要么全部执行,要么全部不执行。
事务有四个特性:ACID,接下来我们一起看看事务的这四个特性
原子性
操作本身不具有原子性,一个操作可以再分为多个操作,当该操作中一步出现错误时,整个操作都无法继续执行,而已经执行的操作可能会造成数据更新的丢失或者错误。
事务的原子性将一系列数据库操作集中在一起,全部要么执行要么不执行,不会出现上述执行一半产生的错误。
持久性
数据库要求数据被写入后能够安全持久地存储在数据库中,当事务被提交后就无法再次回滚,唯一能够撤回已经提交事务的方式是创建一个相反的事务对原操作进行返回,这也是事务持久性的体现
隔离性
隔离性指当多个用户并发访问数据库时,比如操作同一张表时,数据库为每一个用户开启的事务不能被其他事务的操作干扰,多个并发事务之间要相互隔离:举例来说:当a和b同时想要买一张票时,a和b需要对同一张票进行操作,但是不能出现a和b买到同一张票的情况,即a和b对同一张票的状态做了修改。
一致性
在理解数据库的一致性时,会产生让人混淆的状况,原因是数据库中事务的一致性其实包含两个一致性,一个是ACID中的一致性,一个CAP定义的一致性。那我们分别看一下这两个一致性:
ACID一致性:如果一个事务原子地在一个一致的数据库中独立运行,那么在它执行后,数据库的状态一定是一致的。简单来说,一是事务的执行前后不会违背对数据库本身的一些约束比如:完整性约束,主键约束,对数据库的写入操作都是合法的。二是事务在逻辑上是正确的,比如银行转账不可能只扣钱不加钱。
CAP一致性:
分布式系统中各个节点对于同一数据的拷贝具有相同的值
事务的使用:
- START TRANSACTION或BEGIN开始新的事务
- COMMIT提交当前事务,使其更改永久化
- ROLLBACK回滚当前事务,取消之前的操作
- SET autocommit 禁用或者启动当前会话的默认自动提交模式
默认情况下,MySQL在启用自动提交模式的情况下运行。意味着只要执行更新或修改表的语句,MySQL就会将更新持久地存储在磁盘上,无法回滚更改。
如果要显式禁用自动提交模式,可以使用下面语句:
SET autocommit = 0;
接下来我们看一下使用事务的例子:
给goods和employee表中添加数据
BEGIN;
INSERT INTO goods VALUES (5,"apple","food",9);
INSERT INTO employee VALUES (6,"SEOEN"),(7,"MINYOUK");
COMMIT;
最后我们再重点看一下事务的ACID特性中的隔离性
当同时开启多个事务操作数据库中的数据时,数据库系统要对不同的事务进行隔离,保证不同的用户获取数据的准确性。
如果没有隔离会发生的三个问题
脏读
一个事务处理过程中读取了另一个未提交的事务的数据。
不可重复读
对于某个数据,一个事务范围内多次查询却返回了不同的结果,这是由于在查询的间隔,另外的事务修改并提交了数据。
幻读
在一个事务中读取到了别的事务插入的数据,导致前后不一致。
事务的隔离级别以及效果:
举例:
-
在未提交读的隔离等级下,脏读,不可重复读,幻读出现的情况:
事务开始前score表中的数据:
事务1:
BEGIN;
SELECT @@transaction_isolation;
SET SESSION TRANSACTION isolation level read uncommitted;
SELECT @@transaction_isolation;
SELECT * FROM score;
SELECT * FROM score;
COMMIT;
事务2
BEGIN;
SELECT @@transaction_isolation;
UPDATE score SET score = score + 5 WHERE id= 1;
DELETE FROM score WHERE id = 4;
SELECT * FROM score;
执行事务2给score表id为1的成绩加5后结果:
执行事务2删除id为4的数据查询结果:
我们可以看到在上面的例子中,隔离等级设为未提交读,当事务2中操作未提交时就在事务1中读到了事务2操作生效后的数据,而实际上事务2中的操作没有提交也就是还没有更新到数据库中。同时在事务1中两次查询到了不同的结果。出现了脏读和不可重复读的现象。
-
在已提交读的隔离等级下是否会出现脏读,不可重复读的情况:
事务开始前的数据库数据:
事务1
BEGIN;
SELECT @@transaction_isolation;
SET SESSION TRANSACTION isolation level read committed;
SELECT @@transaction_isolation;
SELECT * FROM score;
SELECT * FROM score;
COMMIT;
事务2
BEGIN;
SELECT @@transaction_isolation;
SET SESSION TRANSACTION isolation level read committed;
SELECT @@transaction_isolation;
UPDATE score SET score = score + 5 WHERE id= 1;
SELECT * FROM score;
COMMIT;
打开事务1和2两个事务,将其隔离等级设为已提交读,在事务2中对id为1的成绩加5,在事务2中查询结果加5,事务1中查询结果没变。我们将事务2提交后再在事务1中查询看到结果发生了变化。即在隔离等级为已提交读的情况下不会出现可重复读的现象。