您的位置:首页 > 编程语言 > PHP开发

php文件锁解析

2015-12-16 15:13 627 查看

php文件锁

bool flock() ( int handle, int operation [, int &wouldblock] );

flock() 操作的handle 必须是一个已经打开的文件指针。operation 可以是一下值中的一个:


1. 要取得共享锁(读取程序),将operation 设为 LOCK_SH (php4.0.1以前的版本设置为1)

2. 要取得独占锁(写入程序),将operation 设为 LOCK_XH (php4.0.1以前的版本设置为2)

3. 要释放锁定(无论共享或独占),将operation 设为 LOCK_UN (php4.0.1以前的版本设置为3)

4. 如果你不希望flock()在锁定是堵塞,将operation 设为 LOCK_NB(php4.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);


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 好像选择不了锁,不知道他用的什么锁,和不加锁时得到的结果是一样的,数据不完整。

文件锁有两种:共享锁(LOCK_SH)和排它锁(LOCK_XH),也就是读锁和写锁。

文件锁一般这么使用:

$fp = fopen("filename", "a");
flock($fp, LOCK_SH) or die("lock error")
$str = fread($fp, 1024);
flock($fp, LOCK_UN);
fclose($fp);


fwrite()之后文件立即就被更新了,而不是等到fclose()之后。

读的时候:

如果不想出现dirty数据,最好使用LOCK_SH共享锁。可以考虑一下三种情况:

1. 如果读的时候没有加共享锁,那么其他程序写的话(不管这个写加锁还是不加锁)都会立即写成功。如果正好读了一半,然后被其他程序给写了,那么读的后一半就可能跟前一半对不上(前一半是修改前的,后一半是修改后的)

2. 如果读的时候加上了共享锁(因为只是读,没必要加排它锁),这个时候其他程序开始写,这个写程序没有使用锁,那么写程序会直接修改这个文件,也会导致前面一样的问题。

3. 最理想的情况是,读的时候加锁(LOCK_SH),写的时候也加锁(LOCK_XH),这样写程序会等着读程序完成之后才进行操作,而不会出现贸然操作的情况。

写的时候:

如果多个写程序不加锁同时对文件进行操作,那么最后的数据可能一部分是a程序写的,一部分是b程序写的。

如果写的时候加锁了,这个时候有其他的程序来读,那么他会读到什么东西呢?

1. 如果读程序没有申请共享锁,那么他会读到dirty数据。比如写程序要写a,b,c三部分,写完a,这时候读到的是a,继续写b,这时候读到的是ab,然后写c,这时候读到的是abc

2. 如果读程序之前申请了共享锁,那么读程序会等写程序将abc写完并释放锁之后才进行读。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: