欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  后端开发

php中 flock 文件锁 详解

程序员文章站 2022-04-11 14:01:21
...
  1. $fp = fopen("/tmp/lock.txt", "w+");
  2. if (flock($fp, LOCK_EX)) { // 进行排它型锁定
  3. fwrite($fp, "Write something here ");
  4. flock($fp, LOCK_UN); // 释放锁定
  5. } else {
  6. echo "Couldn't lock the file !";
  7. }
  8. fclose($fp);
  9. ?>
复制代码

备注: 由于 flock() 需要一个文件指针, 因此可能不得不用一个特殊的锁定文件来保护打算通过写模式打开的文件的访问(在 fopen() 函数中加入 "w" 或 "w+")。

注意:flock() 不能用于 NFS 以及其它一些网络文件系统。详细资料查看自己操作系统的文档。 在部分操作系统中 flock() 以进程级实现。当用一个多线程服务器 API(比如 ISAPI)时,可能不可以依靠 flock() 来保护文件,因为运行于同一服务器实例中其它并行线程的 PHP 脚本可以对该文件进行处理。 flock() 不支持旧的文件系统,如 FAT 以及它的派生系统。因此,此环境下总是返回 FALSE(尤其是对 Windows 98 用户来说)。

php中文件锁函数flock函数用法简介:

语法: bool flock ( int $handle , int $operation [, int &$wouldblock ] ) flock() 操作的 handle 必须是一个已经打开的文件指针。operation 可以是以下值之一: 1. 要取得共享锁定(读取程序),将 operation 设为 LOCK_SH(PHP 4.0.1 以前的版本设置为 1) 2. 要取得独占锁定(写入程序),将 operation 设为 LOCK_EX(PHP 4.0.1 以前的版本中设置为 2) 3. 要释放锁定(无论共享或独占),将 operation 设为 LOCK_UN(PHP 4.0.1 以前的版本中设置为 3) 4. 如果你不希望 flock() 在锁定时堵塞,则给 operation 加上 LOCK_NB(PHP 4.0.1 以前的版本中设置为 4)

来看下面的代码。 文件:a.php

  1. $file = “temp.txt”;
  2. $fp = fopen($file , 'w‘);
  3. if(flock($fp , LOCK_EX)){
  4. fwrite($fp , “abc”);
  5. sleep(10);
  6. fwrite($fp , “123”);
  7. flock($fp , LOCK_UN);
  8. }
  9. fclose($fp);
  10. ?>
复制代码

文件:b.php

  1. $file = “temp.txt”;
  2. $fp = fopen($file , 'r');
  3. echo fread($fp , 100);
  4. fclose($fp);
  5. ?>
复制代码

运行 a.php 后,马上运行 b.php ,可以看到输出: abc 等 a.php 运行完后运行 b.php ,可以看到输出: abc 123 显然,当 a.php 写文件时数据太大,导致时间比较长时,这时 b.php 读取数据不完整,在对b.php做修改。 修改 b.php 为:

  1. $file = “temp.txt”;
  2. $fp = fopen($file , 'r‘);
  3. if(flock($fp , LOCK_EX)){
  4. echo fread($fp , 100);
  5. flock($fp , LOCK_UN);
  6. } else{
  7. echo “Lock file failed…”;
  8. }
  9. fclose($fp);
  10. ?>
复制代码

运行 a.php 后,马上运行 b.php ,可以发现 b.php 会等到 a.php 运行完成后(即 10 秒后)才显示: abc 123 读取数据完整,但时间过长,他要等待写锁释放,再对b.php做修改。 修改 b.php 为:

  1. $file = “temp.txt”;
  2. $fp = fopen($file , 'r');
  3. if(flock($fp , LOCK_SH | LOCK_NB)){
  4. echo fread($fp , 100);
  5. flock($fp , LOCK_UN);
  6. } else{
  7. echo “Lock file failed…”;
  8. }
  9. fclose($fp);
  10. ?>
复制代码

运行 a.php 后,马上运行 b.php ,可以看到输出: Lock file failed… 证明可以返回锁文件失败状态,而不是向上面一样要等很久。 脚本之家小编结论: 建议作文件缓存时,选好相关的锁,不然可能导致读取数据不完整,或重复写入数据。 file_get_contents 好像选择不了锁,不知道他默认用的什么锁,反正和不锁得到的输出一样,是不完整的数据。