PHP中读取文件需加锁防并发冲突,方法包括:一、flock()咨询锁;二、排他锁降级为共享锁;三、临时锁文件机制;四、阻塞式flock();五、SplFileObject封装锁逻辑。

如果您在PHP中同时读取和写入同一个文件,多个进程或线程可能造成数据不一致或覆盖丢失,则需要在读取时对文件加锁以防止并发冲突。以下是实现PHP文件读取与锁定的多种方法:
一、使用flock()函数进行 advisory 锁定
flock() 是 PHP 提供的基于文件描述符的轻量级咨询性锁机制,适用于同一文件系统上的进程间协调。该锁在文件关闭或进程终止时自动释放,需配合 fopen() 使用。
1、使用 fopen() 以只读模式打开目标文件,并获取文件指针。
2、调用 flock($fp, LOCK_SH) 对文件施加共享锁(读锁)。
立即学习“PHP免费学习笔记(深入)”;
3、执行 fread() 或 file_get_contents() 读取内容。
4、读取完成后调用 flock($fp, LOCK_UN) 显式释放锁。
5、使用 fclose() 关闭文件指针。
二、在读取前尝试独占写锁并降级为共享锁
当需确保读取期间无其他进程写入,可先申请排他锁(LOCK_EX),成功后再降级为共享锁(LOCK_SH),从而阻塞后续写操作但允许多个读取并发。
1、以 r+ 模式打开文件,获得可读写文件指针。
2、调用 flock($fp, LOCK_EX | LOCK_NB) 尝试非阻塞获取排他锁。
3、若成功,立即调用 flock($fp, LOCK_SH) 转换为共享锁。
4、读取完毕后调用 flock($fp, LOCK_UN) 解锁。
5、关闭文件指针。
三、使用临时锁文件(lockfile)机制
该方法不依赖操作系统内核锁,而是通过创建和删除同名 .lock 文件来模拟互斥访问,适用于 NFS 等不支持 flock() 的环境。
1、构造与目标文件同名的锁文件路径,例如 example.txt.lock。
2、使用 file_put_contents($lock_path, getmypid(), LOCK_EX) 原子写入当前进程 PID 并加锁。
3、检查是否成功写入且内容为当前 PID,确认锁已获取。
4、执行 file_get_contents($target_file) 完成读取。
5、读取完成后 unlink($lock_path) 删除锁文件。
四、使用 stream_set_blocking() 配合 flock() 实现阻塞等待
当多个进程竞争读取时,可让未获锁的进程暂停执行直至锁可用,避免轮询消耗 CPU。
1、以 rb 模式打开文件并获取流资源。
2、调用 stream_set_blocking($fp, true) 确保流处于阻塞模式。
3、调用 flock($fp, LOCK_SH);若锁被占用,PHP 将挂起当前请求直到锁释放。
4、执行读取操作,如 fpassthru($fp) 或 fgets() 循环读取。
5、调用 flock($fp, LOCK_UN) 释放锁并 fclose($fp)。
五、使用 SPL FileObject 类封装锁定逻辑
借助 SplFileObject 可提升代码可读性与复用性,需手动集成 flock() 调用,因其本身不自动加锁。
1、实例化 SplFileObject 并传入文件路径,设置 $useIncludePath = false。
2、调用 $fileObject->getHandle() 获取底层资源句柄。
3、对句柄执行 flock($handle, LOCK_SH) 获取共享锁。
4、调用 $fileObject->rewind() 和 $fileObject->current() 逐行读取。
5、读取结束后调用 flock($handle, LOCK_UN) 并允许对象析构自动关闭句柄。











