• php文件锁


    文件锁:php flock 文件锁.md bool flock ( resource $handle , int $operation [, int &$wouldblock ] )

    flock()允许执行一个简单的可以在任何平台中使用的读取/写入模型(包括大部分的Unix派生版和windows)

    PHP 5.3.2版本之前,锁会被 fclose() 释放(在脚本结束后会自动调用),现在需要手动解锁

    operation

    1. LOCK_SH 取得共享锁(读取的程序)
    2. LOCK_EX 取得独占锁(写入的程序)
    3. LOCK_UN 释放锁定 (无论共享或独占)
    4. LOCK_NB 在flock()锁定时不阻塞

    实例1:A使用独占锁写文件,B读取文件,阻塞

    a.php

    <?php $file = 'test.txt'; $fp = fopen($file, 'w'); if (flock($fp, LOCK_EX)) { // 取得独占锁 fwrite($fp, "Hello World "); // 写入数据 sleep(10); // sleep 10秒,文件被锁定 fwrite($fp, "Hello PHP "); // 写入数据 flock($fp, LOCK_UN); // 解锁 } fclose($fp); ?>

    b.php

    <?php $file = 'test.txt'; $fp = fopen($file, 'r'); if (flock($fp, LOCK_SH)) { // 取得共享锁 while (!feof($fp)) { echo fread($fp, 100); } flock($fp, LOCK_UN); } fclose($fp); ?> 

    先执行a.php,然后执行b.php

    a取得独占锁,b只能等待,等a执行完解除锁定后才能执行b,阻塞

    实例2:A,B都使用共享锁,不阻塞

    a.php

    <?php $file = 'test.txt'; $fp = fopen($file, 'r'); if (flock($fp, LOCK_SH)) { // 取得共享锁 sleep(10); // sleep 10 while (!feof($fp)) { echo fread($fp, 100); } flock($fp, LOCK_UN); } fclose($fp); ?> 

    b.php

    <?php $file = 'test.txt'; $fp = fopen($file, 'r'); if (flock($fp, LOCK_SH)) { // 取得共享锁 while (!feof($fp)) { echo fread($fp, 100); } flock($fp, LOCK_UN); } fclose($fp); ?> 

    先执行a.php,然后执行b.php

    b不需要等待a执行完就能输出文件内容,非阻塞

    实例3:A,B都使用独占锁写文件,阻塞

    a.php

    <?php $file = 'test.txt'; $fp = fopen($file, 'a'); if (flock($fp, LOCK_EX)) { // 取得独占锁 fwrite($fp, "Hello World "); // 写入数据 sleep(10); // sleep 10秒,文件被锁定 fwrite($fp, "Hello PHP "); // 写入数据 flock($fp, LOCK_UN); // 解锁 } fclose($fp); ?>

    b.php

    <?php $file = 'test.txt'; $fp = fopen($file, 'a'); if (flock($fp, LOCK_EX)) { // 取得独占锁 fwrite($fp, "How Are You "); // 写入数据 fwrite($fp, "Show Me The Money "); // 写入数据 flock($fp, LOCK_UN); // 解锁 } fclose($fp); ?>

    先执行a.php,然后执行b.php

    b需要等待a执行完,才能写入数据,阻塞

    实例4:LOCK_NB 锁定时不阻塞,不等待

    a.php

    <?php $file = 'test.txt'; $fp = fopen($file, 'a'); if (flock($fp, LOCK_EX)) { // 取得独占锁 fwrite($fp, "Hello World "); // 写入数据 sleep(10); // sleep 10秒,文件被锁定 fwrite($fp, "Hello PHP "); // 写入数据 flock($fp, LOCK_UN); // 解锁 } fclose($fp); ?>

    b.php

    <?php $file = 'test.txt'; $fp = fopen($file, 'a'); if (flock($fp, LOCK_EX | LOCK_NB)) { // 取得独占锁 fwrite($fp, "How Are You "); // 写入数据 fwrite($fp, "Show Me The Money "); // 写入数据 flock($fp, LOCK_UN); // 解锁 } else { echo 'file locked'; } fclose($fp); ?> 

    先执行a.php,然后执行b.php

    b取不到独占锁,不需要等待a执行完,而是直接返回取不到锁提示,非阻塞

    总结:

    • 使用共享锁LOCK_SH,如果是读取,不需要等待,但如果是写入,需要等待读取完成。
    • 使用独占锁LOCK_EX,无论写入/读取都需要等待。
    • LOCK_UN,无论使用共享/读占锁,使用完后需要解锁。
    • LOCK_NB,当被锁定时,不阻塞,而是提示锁定。
  • 相关阅读:
    又是一个递归的题
    全文检索引擎记录
    处理“System.Web.HttpException: 超过了最大请求长度”
    完整国内城市js级联选择
    多服务器共享session的解决方案之一
    递归求n的阶乘
    面试遇到了 <计算从1到N中1的出现次数>结果悲剧了.
    DropDownList绑定Xml数据的实现
    简单图片上传示例
    C#默认访问修饰符
  • 原文地址:https://www.cnblogs.com/smilevv/p/13261127.html
Copyright © 2020-2023  润新知