php flock独占锁实例及测试

时间:2016-08-16
为了确保文件操作的有效性和完整性,可以通过锁机制将并发状态转换成串行状态,本文章向大家介绍php flock独占锁的使用实例及测试,需要的朋友可以参考一下。

建两个文件:

a.php

<?php
$file = 'test.txt';

$fp = fopen($file, 'w');

if(flock($fp, LOCK_EX)){            // 取得独占锁
    fwrite($fp, "Hello World\r\n"); // 写入数据
    sleep(10);                      // sleep 10秒,文件被锁定
    fwrite($fp, "Hello PHP\r\n");   // 写入数据
    flock($fp, LOCK_UN);            // 解锁
}

fclose($fp);
?>

 

b.php

<?php
$file = 'test.txt';

$fp = fopen($file, 'r');
/* http://www.manongjc.com/article/1360.html */
if(flock($fp, LOCK_SH)){ // 取得贡献锁
    while(!feof($fp)){
        echo fread($fp, 100);
    }
    flock($fp, LOCK_UN);
}

fclose($fp);
?>

运行 a.php 后,马上运行 b.php ,可以看到输出: 

abc 

等 a.php 运行完后运行 b.php ,可以看到输出: 

abc 
123 

显然,当 a.php 写文件时数据太大,导致时间比较长时,这时 b.php 读取数据不完整,在对b.php做修改 

修改 b.php 为:

<?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…”; 
} 
fclose($fp);  
?> 

运行 a.php 后,马上运行 b.php ,可以发现 b.php 会等到 a.php 运行完成后(即 10 秒后)才显示: 

abc 
123 

读取数据完整,但时间过长,他要等待写锁释放,再对b.php做修改。 

修改 b.php 为: 

<?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…”; 
} 
fclose($fp); 
?> 

运行 a.php 后,马上运行 b.php ,可以看到输出: 

Lock file failed… 

证明可以返回锁文件失败状态,而不是向上面一样要等很久。 

结论:

建议作文件缓存时,选好相关的锁,不然可能导致读取数据不完整,或重复写入数据。 

file_get_contents 好像选择不了锁,不知道他默认用的什么锁,反正和不锁得到的输出一样,是不完整的数据。

上一页 下一页

原文地址:http://www.manongjc.com/article/1360.html