php 实现乐观锁实现
乐观锁的概念
乐观锁是一种并发控制机制,假设多用户并发操作时不会产生冲突,因此不加锁。只在数据提交时检查数据是否被其他事务修改过。通常通过版本号(Version)或时间戳(Timestamp)实现。

实现乐观锁的核心步骤
数据库表设计
在表中添加一个版本号字段(如version),每次更新数据时版本号递增。

CREATE TABLE products (
id INT PRIMARY KEY,
name VARCHAR(100),
stock INT,
version INT DEFAULT 0
);
更新数据的逻辑
- 读取数据时获取当前版本号。
- 更新时检查版本号是否与读取时一致,若一致则更新并递增版本号。
// 假设从数据库获取的数据
$productId = 1;
$currentVersion = 5; // 读取时的版本号
$newStock = 10;
// 更新语句,通过版本号校验
$sql = "UPDATE products SET stock = :stock, version = version + 1
WHERE id = :id AND version = :version";
$stmt = $pdo->prepare($sql);
$stmt->bindParam(':stock', $newStock);
$stmt->bindParam(':id', $productId);
$stmt->bindParam(':version', $currentVersion);
$affectedRows = $stmt->execute();
if ($affectedRows === 0) {
// 更新失败,版本号已变更或数据不存在
throw new Exception("并发冲突,数据已被修改");
}
使用时间戳实现乐观锁
若不想用版本号,可用last_updated字段记录时间戳。
ALTER TABLE products ADD COLUMN last_updated TIMESTAMP;
$lastUpdated = '2023-10-01 12:00:00'; // 读取时的时间戳
$sql = "UPDATE products SET stock = :stock, last_updated = NOW()
WHERE id = :id AND last_updated = :lastUpdated";
// 其他逻辑与版本号类似
乐观锁的适用场景
- 读多写少的场景,冲突概率低。
- 需要高并发性能,避免悲观锁的开销。
注意事项
- 冲突时需重试或提示用户重新操作。
- 长时间操作可能因版本号失效导致重试次数增多。
通过版本号或时间戳的比对,可以简单高效地实现乐观锁机制。






