file_put_contents并发性问题解决方案整理

作者:袖梨 2022-06-24


解决 办法一,file_put_contents 高并发与独占锁定

 

发现高并发访问时使用 file_put_contents 写入文件造成数据置空。

查看官方文档:

 

int file_put_contents ( string $filename , string $data [, int $flags [, resource $context ]] )
参数:

 

filename 要被写入数据的文件名。 data 要写入的数据。类型可以是 string,array 或者是 stream 资源(如上面所说的那样)。 flags flags 可以是 FILE_USE_INCLUDE_PATH,FILE_APPEND 和/或 LOCK_EX(获得一个独占锁定),然而使用 FILE_USE_INCLUDE_PATH 时要特别谨慎。 context 一个 context 资源。
直接直至 flags 参数为 LOCK_EX 即可在高并发时获得一个独占锁定。

另外,flock 函数的也提供了文件锁定方法:

 
$fp = fopen("/tmp/lock.txt", "w+");
if (flock($fp, LOCK_EX)) { // 进行排它型锁定
fwrite($fp, "Write something heren");
flock($fp, LOCK_UN); // 释放锁定
} else {
echo "Couldn't lock the file !";
}
fclose($fp);

 

注意 flock() 需要一个文件指针。

 

方法二,file_put_contents() 存在高并发问题,Smarty中处理方式如下。

<?php

 

define("FILE_PUT_CONTENTS_ATOMIC_TEMP", dirname(__FILE__)."/cache");

define("FILE_PUT_CONTENTS_ATOMIC_MODE", 0777);

 

function file_put_contents_atomic($filename, $content) {

 

    $temp = tempnam(FILE_PUT_CONTENTS_ATOMIC_TEMP, 'temp');

    if (!($f = @fopen($temp, 'wb'))) {

        $temp = FILE_PUT_CONTENTS_ATOMIC_TEMP . DIRECTORY_SEPARATOR . uniqid('temp');

        if (!($f = @fopen($temp, 'wb'))) {

            trigger_error("file_put_contents_atomic() : error writing temporary file '$temp'", E_USER_WARNING);

            return false;

        }

    }

 

    fwrite($f, $content);

    fclose($f);

 

    if (
!@rename($temp
, $filename)) {

        @unlink($filename);

        @rename($temp, $filename);

    }

 

    @chmod($filename, FILE_PUT_CONTENTS_ATOMIC_MODE);

 

    return true;

 

}

 

?>

相关文章

精彩推荐