flock()函数用于php文件锁,防止多进程读写冲突。通过LOCK_EX、LOCK_SH等类型实现排他或共享锁,结合LOCK_NB可非阻塞尝试,循环与usleep可模拟超时;锁在进程结束或fclose时自动释放,但建议显式解锁;适用于单机环境的定时任务防重、计数器更新等场景,分布式需用redis等替代;注意避免死锁、确保操作原子性,windows与linux平台行为一致。

文件锁在PHP中主要用于防止多个进程同时读写同一个文件导致数据错乱。PHP通过flock()函数提供文件锁定机制,能有效实现文件的并发控制。正确使用文件锁可以避免资源竞争,但若使用不当也可能引发死锁或性能问题。
1. flock() 函数基本用法
flock() 是PHP操作文件锁的核心函数,它作用于文件句柄,支持多种锁类型:
- LOCK_SH:共享锁(读锁),多个进程可同时持有
- LOCK_EX:排他锁(写锁),仅一个进程可持有
- LOCK_UN:释放锁
- LOCK_NB:非阻塞模式,与其他锁类型配合使用
示例:使用排他锁写入文件
$fp = fopen("log.txt", "a"); if (flock($fp, LOCK_EX)) { fwrite($fp, "记录一条日志n"); flock($fp, LOCK_UN); // 释放锁 } else { echo "无法获取锁"; } fclose($fp);
2. 非阻塞锁与超时处理
默认情况下,flock() 在无法获得锁时会阻塞等待。可通过 LOCK_NB 实现非阻塞尝试:
立即学习“PHP免费学习笔记(深入)”;
$fp = fopen("data.txt", "r+"); if (flock($fp, LOCK_EX | LOCK_NB)) { // 成功获取锁,进行写操作 ftruncate($fp, 0); fwrite($fp, "新数据"); flock($fp, LOCK_UN); } else { echo "文件正被占用,稍后重试"; } fclose($fp); </font> <p>如需实现超时等待,可结合循环和sleep:</p> <div class="aritcle_card"> <a class="aritcle_card_img" href="/ai/%E5%B7%A7%E6%96%87%E4%B9%A6"> <img src="https://img.php.cn/upload/ai_manual/001/246/273/68b6c52f05eb9477.png" alt="巧文书"> </a> <div class="aritcle_card_info"> <a href="/ai/%E5%B7%A7%E6%96%87%E4%B9%A6">巧文书</a> <p>巧文书是一款AI写标书、AI写方案的产品。通过自研的先进AI大模型,精准解析招标文件,智能生成投标内容。</p> <div class=""> <img src="/static/images/card_xiazai.png" alt="巧文书"> <span>8</span> </div> </div> <a href="/ai/%E5%B7%A7%E6%96%87%E4%B9%A6" class="aritcle_card_btn"> <span>查看详情</span> <img src="/static/images/cardxiayige-3.png" alt="巧文书"> </a> </div> <font size="2"> <pre class="brush:php;toolbar:false;"> $attempts = 0; while (!flock($fp, LOCK_EX | LOCK_NB) && $attempts < 5) { usleep(200000); // 等待200ms $attempts++; }
3. 使用注意事项
使用PHP文件锁时需注意以下关键点:
- 锁是基于进程的:脚本结束或调用 fclose() 时自动释放锁,无需手动解锁也能释放,但仍建议显式释放
- 只对同一台服务器有效:NFS等网络文件系统可能不支持flock,分布式环境需使用redis或数据库锁
- 避免死锁:按固定顺序加锁,及时释放,长时间操作应尽量减少持锁时间
- 锁不保证原子性:flock() 后的读写仍需确保逻辑正确,尤其是大文件操作
- windows和Linux行为一致:PHP抽象了系统差异,flock跨平台可用
4. 实际应用场景
常见用途包括:
- 定时任务防重复执行:多个cron任务竞争同一个锁文件
- 计数器更新:避免并发写导致计数错误
- 缓存重建:防止多个请求同时重建缓存
- 日志写入:确保多进程写日志不混乱
例如防重复执行:
$lockFile = fopen("/tmp/update.lock", "w"); if (!flock($lockFile, LOCK_EX | LOCK_NB)) { die("已有进程在运行"); } // 执行任务... flock($lockFile, LOCK_UN); fclose($lockFile);
基本上就这些。合理使用flock()能解决大多数单机场景下的并发问题,关键是理解其作用范围和局限性。不复杂但容易忽略细节。