SELECT ... FOR UPDATE 是 SQL 中的一个重要句型,特别是在使用 MySQL 进行事务处理时。这个句型允许你在事务中锁定选中的行,以防止其他事务对这些行进行更新或删除,直到当前事务结束(提交或回滚)。这对于保持数据的一致性和完整性特别有用,特别是在处理涉及多个步骤的操作时。
基本语法
START TRANSACTION; -- 开始一个事务
SELECT column_name(s)
FROM table_name
WHERE condition
FOR UPDATE;
-- 这里执行更新操作
COMMIT; -- 提交事务
-- 或者
ROLLBACK; -- 回滚事务
使用场景
- 银行转账:从一个账户转移到另一个账户时,需要确保两个账户的余额在转账过程中不会被其他事务修改。
- 库存更新:在电子商务系统中,当你从库存中减去一个商品的数量时,需要确保没有其他事务也在尝试修改这个数量。
注意事项
- 锁定范围:
FOR UPDATE会锁定满足WHERE子句条件的所有行。如果WHERE子句没有指定条件,那么会锁定整个表。 - 隔离级别:不同的数据库事务隔离级别会影响
FOR UPDATE的行为。例如,在READ UNCOMMITTED隔离级别下,FOR UPDATE可能不会按预期工作。 - 死锁:当两个或多个事务相互等待对方释放锁定的资源时,就会发生死锁。在使用
FOR UPDATE时,需要注意避免死锁的发生。 - 性能影响:锁定行会影响数据库的并发性能,因为它限制了其他事务对这些行的访问。因此,应该谨慎使用
FOR UPDATE,并确保只在必要时使用。 - InnoDB 存储引擎:
FOR UPDATE只在支持行级锁定的存储引擎(如 InnoDB)中有效。MyISAM 存储引擎不支持行级锁定,因此FOR UPDATE在 MyISAM 表中不会按预期工作。
示例
假设有一个名为 accounts 的表,包含 id 和 balance 两个字段,想从账户 A 转账到账户 B,同时确保这两个账户的余额在转账过程中不会被其他事务修改:
START TRANSACTION;
-- 锁定账户A和账户B的余额
SELECT balance FROM accounts WHERE id = 1 FOR UPDATE;
SELECT balance FROM accounts WHERE id = 2 FOR UPDATE;
-- 检查余额,执行转账逻辑...
COMMIT;
在这个例子中,FOR UPDATE 确保了在转账过程中,账户 A 和账户 B 的余额不会被其他事务修改。
361

被折叠的 条评论
为什么被折叠?



