解决办法一,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 = <a href="/"/tags.php/fopen//"" target="/"_blank/"">fopen</a>("/tmp/lock.txt", "w+");
- if (flock($fp, LOCK_EX)) { // 进行排它型锁定
- fwrite($fp, "Write something here//n");
- 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;
- }
- ?>
新闻热点
疑难解答