php锁实现
PHP 锁的实现方法
文件锁(flock)
文件锁是一种基于文件的互斥锁实现方式,适用于单机环境。通过 flock 函数可以对文件进行加锁和解锁操作。
$fp = fopen("lockfile.lock", "w+");
if (flock($fp, LOCK_EX)) { // 获取独占锁
// 执行需要加锁的代码
flock($fp, LOCK_UN); // 释放锁
} else {
echo "无法获取锁";
}
fclose($fp);
LOCK_EX 表示独占锁,其他进程无法获取;LOCK_SH 表示共享锁,允许多个进程同时读取。文件锁在进程结束后会自动释放,但显式调用 LOCK_UN 更安全。
数据库锁
数据库锁利用数据库的原子性操作实现,适用于分布式环境。常见方式包括行锁、表锁或乐观锁。
// 使用 SELECT ... FOR UPDATE 实现行锁
$pdo->beginTransaction();
$stmt = $pdo->prepare("SELECT * FROM orders WHERE id = 1 FOR UPDATE");
$stmt->execute();
// 执行需要加锁的操作
$pdo->commit();
乐观锁通过版本号或时间戳实现:
// 更新时检查版本号
$stmt = $pdo->prepare("UPDATE products SET stock = stock - 1, version = version + 1 WHERE id = 1 AND version = ?");
$stmt->execute([$oldVersion]);
if ($stmt->rowCount() === 0) {
throw new Exception("并发冲突");
}
Redis 锁
Redis 锁适用于分布式系统,通过 SETNX 或 RedLock 算法实现。
$redis = new Redis();
$redis->connect('127.0.0.1', 6379);
$lockKey = 'resource_lock';
$token = uniqid();
$expire = 300; // 锁过期时间(秒)
// 尝试获取锁
if ($redis->set($lockKey, $token, ['NX', 'EX' => $expire])) {
try {
// 执行需要加锁的代码
} finally {
// 释放锁(Lua脚本保证原子性)
$script = "
if redis.call('get', KEYS[1]) == ARGV[1] then
return redis.call('del', KEYS[1])
else
return 0
end";
$redis->eval($script, [$lockKey, $token], 1);
}
} else {
echo "获取锁失败";
}
Semaphore 信号量
PHP 的 sysvsem 扩展提供 System V 信号量支持,适用于多进程同步。
$semKey = ftok(__FILE__, 't');
$semId = sem_get($semKey);
if (sem_acquire($semId)) { // 阻塞获取信号量
// 执行需要同步的代码
sem_release($semId); // 释放信号量
}
注意事项
- 锁的粒度应尽量小,避免长时间持有锁
- 分布式锁需设置合理的过期时间,防止死锁
- 考虑锁的重入问题(同一线程多次获取锁)
- 文件锁在 NFS 文件系统上可能不可靠
- 数据库锁可能影响性能,需合理设计索引







