bool flock ( int handle, int operation [, int &wouldblock] );
flock() 操作的 handle 必须是一个已经打开的文件指针。operation 可以是以下值之一:
建两个文件
(1) 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 ); |
(2) b.php
$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 ); |
运行 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 ); |
运行 a.php 后,马上运行 b.php ,可以看到输出:
Lock file failed…
证明可以返回锁文件失败状态,而不是向上面一样要等很久。
结论:
建议作文件缓存时,选好相关的锁,不然可能导致读取数据不完整,或重复写入数据。
file_get_contents 好像选择不了锁,不知道他默认用的什么锁,反正和不锁得到的输出一样,是不完整的数据。
我是要做文件缓存,所以只需要知道是否有写锁存在即可,有的话就查数据库就可以了。
测试环境:Linux(Ubuntu 6) , PHP 5.1.2 , Apache 2
再转:
原文地址:http://www.cnblogs.com/henairan/p/3858754.html