MySQL 事务是数据库管理系统 (DBMS) 中的一项关键功能,确保一系列数据库操作作为一个整体被执行,且具有原子性、一致性、隔离性和持久性 (ACID) 的特性。事务处理机制可以帮助开发者确保数据的完整性和一致性,特别是在出现错误或并发操作时。
1. 事务的四大特性 (ACID)
-
原子性 (Atomicity)
事务中的所有操作要么全部完成,要么全部不完成。原子性确保了事务的所有操作作为一个不可分割的单元执行。 -
一致性 (Consistency)
事务从一个一致状态转换到另一个一致状态,即在事务执行前后,数据库的完整性约束不会被破坏。 -
隔离性 (Isolation)
多个事务同时执行时,一个事务的操作不会影响其他事务。隔离性决定了一个事务如何对其他事务的并发执行可见。 -
持久性 (Durability)
一旦事务提交,其结果应永久存储在数据库中,即使系统发生崩溃,提交后的数据也不会丢失。
2. MySQL 中的事务控制语句
MySQL 使用 InnoDB 引擎来支持事务处理。常用的事务控制语句包括:
- START TRANSACTION:显式开启一个事务。
- COMMIT:提交事务,所有对数据库的修改会永久保存。
- ROLLBACK:回滚事务,撤销自事务开启以来的所有修改。
- SAVEPOINT:设置事务的保存点,允许在事务中回滚到特定的保存点。
- RELEASE SAVEPOINT:删除某个保存点,之后无法再回滚到该保存点。
- ROLLBACK TO SAVEPOINT:回滚到某个保存点,之后的修改被撤销,但保存点之前的操作不受影响。
3. MySQL 事务的基本使用
3.1 开启事务
通过 START TRANSACTION
开启事务:
START TRANSACTION;
-- 执行一系列操作
INSERT INTO accounts (user_id, balance) VALUES (1, 1000);
UPDATE accounts SET balance = balance - 100 WHERE user_id = 1;
-- 提交事务
COMMIT;
如果某个操作发生错误,可以通过 ROLLBACK
来回滚整个事务:
START TRANSACTION;
-- 执行操作
INSERT INTO accounts (user_id, balance) VALUES (1, 1000);
-- 出现错误,回滚
ROLLBACK;
3.2 使用保存点
保存点允许你在事务中部分回滚,而不影响之前的操作。
START TRANSACTION;
-- 设置保存点
SAVEPOINT sp1;
INSERT INTO accounts (user_id, balance) VALUES (1, 1000);
-- 设置另一个保存点
SAVEPOINT sp2;
UPDATE accounts SET balance = balance - 100 WHERE user_id = 1;
-- 回滚到 sp1,撤销 sp2 之后的操作
ROLLBACK TO SAVEPOINT sp1;
-- 提交事务
COMMIT;
4. MySQL 事务的隔离级别
事务的隔离性确保多个事务并发执行时不会互相干扰,MySQL 提供了四种隔离级别,不同的隔离级别决定了事务如何处理并发操作的可见性。隔离级别越高,并发性能越低,但数据一致性越强。
MySQL 支持以下四种事务隔离级别:
-
READ UNCOMMITTED(未提交读)
允许一个事务读取另一个未提交事务的修改。会导致“脏读”问题。 -
READ COMMITTED(提交读)
只允许一个事务读取另一个已经提交的事务修改。避免脏读,但可能导致“不可重复读”问题。 -
REPEATABLE READ(可重复读)(MySQL 默认的隔离级别)
保证在同一个事务中多次读取同一数据时,结果一致。避免脏读和不可重复读,但可能导致“幻读”问题。 -
SERIALIZABLE(可串行化)
强制事务顺序执行,完全避免并发问题,但性能较差。避免脏读、不可重复读和幻读。
5. 事务隔离级别的配置
-
查看当前隔离级别:
SELECT @@transaction_isolation;
-
设置全局或会话隔离级别:
-
设置全局隔离级别:
SET GLOBAL transaction_isolation = 'REPEATABLE READ';
-
设置会话隔离级别:
SET SESSION transaction_isolation = 'READ COMMITTED';
-
6. 并发事务中的问题
并发事务可能会导致以下几种问题:
-
脏读
事务A读取了事务B修改的未提交数据,如果事务B回滚,事务A读取到的数据就会失效。 -
不可重复读
事务A在两次读取同一数据时,读到不同的结果。事务B在事务A第二次读取之前,修改了该数据并提交。 -
幻读
事务A读取了满足某条件的多行数据,事务B在事务A读取后插入或删除了一些数据,导致事务A再查询时看到不一致的结果。
7. 示例:处理事务的不同隔离级别
假设我们有两个事务:
- 事务 A 开启时读取一笔数据。
- 事务 B 在 A 进行读取后修改了该数据并提交。
7.1 READ UNCOMMITTED
事务 A 读取未提交的数据:
-- 事务 B
START TRANSACTION;
UPDATE accounts SET balance = 2000 WHERE user_id = 1;
-- 事务 A(读取事务 B 未提交的数据)
SELECT balance FROM accounts WHERE user_id = 1;
-- 事务 B 回滚
ROLLBACK;
在 READ UNCOMMITTED 隔离级别下,事务 A 能够读取到事务 B 未提交的修改,导致脏读问题。
7.2 REPEATABLE READ(默认隔离级别)
在可重复读隔离级别下,事务 A 能够保证多次读取的数据是一致的:
-- 事务 A
START TRANSACTION;
SELECT balance FROM accounts WHERE user_id = 1;
-- 事务 B
START TRANSACTION;
UPDATE accounts SET balance = 2000 WHERE user_id = 1;
COMMIT;
-- 事务 A 再次读取
SELECT balance FROM accounts WHERE user_id = 1;
-- 事务 A 提交
COMMIT;
在 REPEATABLE READ 下,事务 A 的两次读取会得到相同的数据,即使事务 B 已经提交了修改。
8. 使用案例
在电商系统中,事务常用于处理订单、支付、库存等操作。例如,处理用户购买商品时,库存的扣减和订单生成必须是原子操作,如果扣减库存失败,订单应该回滚。这可以通过事务来保证一致性:
START TRANSACTION;
-- 扣减库存
UPDATE products SET stock = stock - 1 WHERE product_id = 1001 AND stock > 0;
-- 生成订单
INSERT INTO orders (user_id, product_id, amount) VALUES (1, 1001, 1);
COMMIT;
如果库存不足或订单生成失败,整个事务将回滚,确保数据一致性。
总结
MySQL 事务通过 ACID 特性确保了数据库操作的可靠性。不同的隔离级别在性能和数据一致性之间做了权衡,开发者应根据具体场景选择适合的事务隔离级别。在事务处理中,掌握回滚、保存点和并发控制等机制是确保数据一致性和完整性的关键。
标签:事务,读取,--,mysql,balance,id,隔离 From: https://blog.csdn.net/m0_73163793/article/details/142099896