如何将数据库事务控制机制为一个?
摘要:事务的四大特性(ACID) 1.1 原子性(Atomicity) 原子性说的是事务中的操作要么一起成功,要么一起失败,事务提交(commit)和事务回滚(rollback)。 1.2 一致
事务的四大特性(ACID)
1.1原子性(Atomicity)
原子性说的是事务中的操作要么一起成功,要么一起失败,事务提交(commit)和事务回滚(rollback)。
1.2一致性(Consistency)
一致性主要说明的是事务的前后,数据库中的数据的状态要确保一致。
事务提交成功,那么张三账户上的余额是900元,李四账户上的余额是100元。
事务提交失败,那么张三和李四的账户的金额不变。
这说明现在在数据库的事务的控制下,确保了数据的一致性。
1.3隔离性(Isolation)
隔离性的体现,多个并发事务之间是隔离的。
1.4持久性(Durability)
持久性的体现就是数据一旦commit之后,那么对于数据的改变就是永久的。
概念说明
以下几个概念是事务隔离级别要实际解决的问题,所以需要搞清楚都是什么意思。
脏读
脏读指的是读到了其他事务未提交的数据,未提交意味着这些数据可能会回滚,可能最终不会存到数据库中,也就是不一定存在的数据。读到了并不一定最终存在的数据,这就是脏读。
可重复读
可重复读指的是在一个事务内,最开始读到的数据和事务结束前的任意时刻读到的同一批数据都是一致的。通常针对数据**更新(UPDATE)**操作。
事务1:通过以下 SQL 语句执行:
START TRANSACTION;
SELECT balance FROM bank_account WHERE account_number = 'A123456';
-- 在此期间,其他事务执行了一次UPDATE操作,将balance修改为2000.00
SELECT balance FROM bank_account WHERE account_number = 'A123456';
COMMIT;
事务2:通过以下 SQL 语句执行:
UPDATE bank_account SET balance = 2000.00 WHERE account_number = 'A123456';
在事务1中,即使在两次SELECT之间发生了事务2的UPDATE操作,因为事务1处于可重复读隔离级别下,两次SELECT都会返回相同的值(1000.00),即第一次SELECT读到的值与第二次SELECT读到的值相同,这是因为事务1始终保持了之前读取到的数据快照。
不可重复读
对比可重复读,不可重复读指的是在同一事务内,不同的时刻读到的同一批数据可能是不一样的,可能会受到其他事务的影响,比如其他事务改了这批数据并提交了。通常针对数据**更新(UPDATE)**操作。
幻读
幻读是针对数据**插入(INSERT)**操作来说的。假设事务A对某些行的内容作了更改,但是还未提交,此时事务B插入了与事务A更改前的记录相同的记录行,并且在事务A提交之前先提交了,而这时,在事务A中查询,会发现好像刚刚的更改对于某些数据未起作用,但其实是事务B刚插入进来的,让用户感觉很魔幻,感觉出现了幻觉,这就叫幻读。
现在,有两个并发事务:
事务A:执行以下操作:
START TRANSACTION;
SELECT * FROM orders WHERE status = 'pending';
-- 在此期间,事务B执行了 INSERT INTO orders (id, order_number, status) VALUES (101, '123456', 'pending');
SELECT * FROM orders WHERE status = 'pending';
COMMIT;
事务B:在事务A的两次SELECT之间执行了以下操作:
INSERT INTO orders (id, order_number, status) VALUES (101, '123456', 'pending');
在这个示例中,事务A在两次SELECT之间,事务B插入了一条新的订单数据。在事务A中的第二次SELECT操作中,会发现了事务B插入的新订单数据,尽管在第一次SELECT中并没有看到这条数据。这种现象就被称为幻读,因为在同一个事务中,看到了好像刚刚新增的数据,就好像出现了幻觉一样。
其中不可重复读主要涉及到同一行数据的更新操作,而幻读主要涉及到数据插入操作。在实际应用中,不可重复读和幻读可能会同时发生,取决于并发事务的具体操作和隔离级别的设置。
事务隔离级别
事务隔离其实就是为了解决上面提到的脏读、不可重复读、幻读这几个问题,SQL 标准定义了四种隔离级别,MySQL 全都支持。
