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
-
- $file = "temp.txt";
- $fp = fopen($file , 'w');
- if(flock($fp , LOCK_EX)){
- fwrite($fp , "abc\n");
- sleep(10);
- fwrite($fp , "123\n");
- flock($fp , LOCK_UN);
- }
- fclose($fp);
$file = "temp.txt";
$fp = fopen($file , 'w');
if(flock($fp , LOCK_EX)){
fwrite($fp , "abc\n");
sleep(10);
fwrite($fp , "123\n");
flock($fp , LOCK_UN);
}
fclose($fp);
b.php
-
- $file = "temp.txt";
- $fp = fopen($file , 'r');
- echo fread($fp , 100);
- fclose($fp);
$file = "temp.txt";
$fp = fopen($file , 'r');
echo fread($fp , 100);
fclose($fp);
运行 a.php 后,马上运行 b.php ,可以看到输出:
abc
等 a.php 运行完后运行 b.php ,可以看到输出:
abc
123
显然,当 a.php 写文件时数据太大,导致时间比较长时,这时 b.php 读取数据不完整
修改 b.php 为:
- $file = "temp.txt";
- $fp = fopen($file , 'r');
- if(flock($fp , LOCK_EX)){
- echo fread($fp , 100);
- flock($fp , LOCK_UN);
- } else{
- echo "Lock file failed...\n";
- }
- fclose($fp);
$file = "temp.txt";
$fp = fopen($file , 'r');
if(flock($fp , LOCK_EX)){
echo fread($fp , 100);
flock($fp , LOCK_UN);
} else{
echo "Lock file failed...\n";
}
fclose($fp);
运行 a.php 后,马上运行 b.php ,可以发现 b.php 会等到 a.php 运行完成后(即 10 秒后)才显示:
abc
123
读取数据完整,但时间过长,他要等待写锁释放。
修改 b.php 为:
- $file = "temp.txt";
- $fp = fopen($file , 'r');
- if(flock($fp , LOCK_SH | LOCK_NB)){
- echo fread($fp , 100);
- flock($fp , LOCK_UN);
- } else{
- echo "Lock file failed...\n";
- }
- fclose($fp);
$file = "temp.txt";
$fp = fopen($file , 'r');
if(flock($fp , LOCK_SH | LOCK_NB)){
echo fread($fp , 100);
flock($fp , LOCK_UN);
} else{
echo "Lock file failed...\n";
}
fclose($fp);
运行 a.php 后,马上运行 b.php ,可以看到输出:
Lock file failed…
证明可以返回锁文件失败状态,而不是向上面一样要等很久。
结论:
建议作文件缓存时,选好相关的锁,不然可能导致读取数据不完整,或重复写入数据。
file_get_contents 好像选择不了锁,不知道他默认用的什么锁,反正和不锁得到的输出一样,是不完整的数据。