php实现自旋锁
PHP 实现自旋锁的方法
自旋锁是一种忙等待的锁机制,线程在获取锁失败时会不断循环尝试,直到成功获取锁。PHP 中可以通过以下几种方式实现自旋锁。
使用文件锁(flock)
通过 PHP 的 flock 函数可以实现基于文件的自旋锁。flock 提供独占锁(LOCK_EX)和非阻塞模式(LOCK_NB)选项。

$lockFile = fopen('lockfile.lock', 'w+');
while (!flock($lockFile, LOCK_EX | LOCK_NB)) {
usleep(100000); // 等待 100 毫秒后重试
}
// 临界区代码
flock($lockFile, LOCK_UN);
fclose($lockFile);
使用共享内存(shmop)
通过 shmop 扩展可以实现基于共享内存的自旋锁。共享内存适合多进程间的同步。

$shmKey = ftok(__FILE__, 't');
$shmId = shmop_open($shmKey, 'c', 0644, 1);
while (shmop_read($shmId, 0, 1) != 0) {
usleep(100000);
}
shmop_write($shmId, '1', 0);
// 临界区代码
shmop_write($shmId, '0', 0);
shmop_close($shmId);
使用 Redis 实现分布式自旋锁
Redis 的 SETNX 命令可以实现分布式自旋锁,适合多服务器环境。
$redis = new Redis();
$redis->connect('127.0.0.1', 6379);
$lockKey = 'spin_lock';
while (!$redis->setnx($lockKey, 1)) {
usleep(100000);
}
$redis->expire($lockKey, 10); // 设置超时防止死锁
// 临界区代码
$redis->del($lockKey);
使用 PHP 的扩展(如 pthreads)
如果使用 pthreads 扩展(仅限 CLI 环境),可以通过 Mutex 类实现自旋锁。
$mutex = new Mutex();
while (!$mutex->trylock()) {
usleep(100000);
}
// 临界区代码
$mutex->unlock();
注意事项
- 避免死锁:确保锁最终会被释放,可以通过设置超时或异常处理实现。
- 性能开销:自旋锁会占用 CPU 资源,需合理设置等待时间(如
usleep)。 - 分布式环境:Redis 或数据库锁更适合跨进程或跨服务器的同步需求。
以上方法根据场景选择,单机环境推荐文件锁或共享内存,分布式环境推荐 Redis 实现。






