>  기사  >  백엔드 개발  >  PHP에서 파일을 동시에 읽고 쓰는 문제를 해결하는 방법

PHP에서 파일을 동시에 읽고 쓰는 문제를 해결하는 방법

小云云
小云云원래의
2018-03-16 15:31:562459검색

동시성이 높은 경우 동일한 파일을 조작하면 데이터 혼란이 발생하므로 파일 조작 시 몇 가지 특별한 처리가 필요합니다. 그것이 모두에게 도움이 되기를 바랍니다.

옵션 1: 무리 기능을 사용하여 파일 잠그기

<span style="font-size: 14px;">/*  <br/>* flock(file,lock,block)  <br/>* file 必需,规定要锁定或释放的已打开的文件  <br/>* lock 必需。规定要使用哪种锁定类型。  <br/>* block 可选。若设置为 1 或 true,则当进行锁定时阻挡其他进程。  <br/>* lock  <br/>* LOCK_SH 要取得共享锁定(读取的程序)  <br/>* LOCK_EX 要取得独占锁定(写入的程序)  <br/>* LOCK_UN 要释放锁定(无论共享或独占)  <br/>* LOCK_NB 如果不希望 flock() 在锁定时堵塞  <br/>/*  <br/><br/>// 获取锁if (flock($file,LOCK_EX)) {    // 操作文件<br/>    fwrite($file,&#39;write more words&#39;);  <br/>    // 操作完毕后释放锁<br/>    flock($file,LOCK_UN);  <br/>} else {  <br/>    //处理错误逻辑  <br/>}  <br/>fclose($file);  <br/></span>

무리 기능은 종종 여러 동시성 상황에서 리소스를 독점하는 것처럼 보이며 즉시 해제하지 않거나 전혀 해제하지 않아 문제가 발생하는 경우가 많습니다. 교착 상태로 인해 교착 상태가 발생합니다. 서버의 CPU 사용량이 매우 높으며 때로는 서버가 완전히 종료되는 경우도 있습니다. 따라서 단순히 무리를 사용하여 잠그는 것만으로는 문제를 완전히 해결할 수 없습니다.

옵션 2: 잠금 시간을 제한하고 시간 초과 시 종료

<span style="font-size: 14px;">if($fp = fopen($fileName,&#39;a&#39;)) {    $startTime = microtime();    do{        $canWrite = flock($fp, LOCK_EX);        if(!$canWrite) {<br/>            usleep(round(rand(0, 100) * 1000));   // 释放cpu,将cpu资源先让给其他进程<br/>        }        // 如果未获取到锁,且未超时,则继续获取锁<br/>    } while((!$canWrite) && ((microtime() - $startTime) < 1000));    if($canWrite) {        fwrite($fp, $dataToSave);<br/>        flock($file,LOCK_UN);  <br/>    }    fclose($fp);<br/>}<br/></span>

옵션 3: 임시 파일 사용

<span style="font-size: 14px;">$dir_fileopen=&#39;tmp&#39;;function cfopen($filename,$mode){<br/>    global $dir_fileopen;<br/>    clearstatcache();    // 创建一个临时文件<br/>    do{        $id=uniqid();        $tempfilename=$dir_fileopen.&#39;/&#39;.$id.md5($filename);<br/>    } while(file_exists($tempfilename));    // 将要操作的文件内容拷贝到临时文件中<br/>    copy($filename,$tempfilename);    $fp = fopen($tempfilename, $mode);    return $fp ? [$fp, $filename, $id, @filemtime($filename)] : false;<br/>}function cfwrite($fp,$string){<br/>    // 将新增内容写入到临时文件中<br/>    return fwrite($fp[0], $string);<br/>}function cfclose($fp){<br/>    global $dir_fileopen;    $success = fclose($fp[0]);<br/>    clearstatcache();    $tempfilename = $dir_fileopen.&#39;/&#39;.$fp[2].md5($fp[1]);    // 如果要操作的文件在操作期间没有被修改过,则说明没有人操作过该文件,那么将临时文件改名为真正的文件<br/>    if((@filemtime($fp[1]) == $fp[3])){<br/>        rename($tempfilename,$fp[1]);<br/>    }else{        //说明有其它进程在操作目标文件,当前进程被拒绝,删除临时文件<br/>        unlink($tempfilename);        $success = false;<br/>    }    return $success;<br/>}$startTime = microtime();do{    $fp=cfopen(&#39;lock.txt&#39;,&#39;a+&#39;);<br/>    cfwrite($fp,"welcome to beijing.\n");    $success = cfclose($fp, &#39;on&#39;);    if(!$success) {<br/>        usleep(round(rand(0, 100) * 1000));   // 释放cpu,将cpu资源先让给其他进程<br/>    }<br/>}while(!$success && ((microtime() - $startTime) < 1000));    // 如果为false,说明操作失败,则重新进行一次操作<br/></span>

옵션 4: 대기열 사용

파일 작업을 위한 대기열을 생성한 후 쓰기 a 스크립트는 대기열에서 파일 작업 정보를 순차적으로 읽고 파일에 대해 해당 작업을 수행합니다. 이런 방식으로 한 번에 하나의 프로세스만 파일을 작업하므로 동시성 문제가 해결됩니다.

위 내용은 http://blog.csdn.net/daiyan_csdn/article/details/51524781

에서 참조했습니다. 동시성이 높은 경우 동일한 파일을 조작하면 데이터 혼란이 발생하므로, 작동해야 합니다. 파일에 대해 일부 특수 처리가 수행됩니다. 몇 가지 해결 방법이 아래에 요약되어 있습니다.

옵션 1: 무리 기능을 사용하여 파일 잠그기

<span style="font-size: 14px;">/*  <br/>* flock(file,lock,block)  <br/>* file 必需,规定要锁定或释放的已打开的文件  <br/>* lock 必需。规定要使用哪种锁定类型。  <br/>* block 可选。若设置为 1 或 true,则当进行锁定时阻挡其他进程。  <br/>* lock  <br/>* LOCK_SH 要取得共享锁定(读取的程序)  <br/>* LOCK_EX 要取得独占锁定(写入的程序)  <br/>* LOCK_UN 要释放锁定(无论共享或独占)  <br/>* LOCK_NB 如果不希望 flock() 在锁定时堵塞  <br/>/*  <br/><br/>// 获取锁if (flock($file,LOCK_EX)) {    // 操作文件<br/>    fwrite($file,&#39;write more words&#39;);  <br/>    // 操作完毕后释放锁<br/>    flock($file,LOCK_UN);  <br/>} else {  <br/>    //处理错误逻辑  <br/>}  <br/>fclose($file);  <br/></span>

무리 기능은 종종 여러 동시성 상황에서 리소스를 독점하는 것처럼 보이며 즉시 해제하지 않거나 전혀 해제하지 않아 교착 상태가 발생합니다. , 이로 인해 서버가 CPU 사용량이 매우 높고 때로는 서버가 완전히 종료되는 경우도 있습니다. 따라서 단순히 무리를 사용하여 잠그는 것만으로는 문제를 완전히 해결할 수 없습니다.

옵션 2: 잠금 시간을 제한하고 시간 초과 시 종료

<span style="font-size: 14px;">if($fp = fopen($fileName,&#39;a&#39;)) {    $startTime = microtime();    do{        $canWrite = flock($fp, LOCK_EX);        if(!$canWrite) {<br/>            usleep(round(rand(0, 100) * 1000));   // 释放cpu,将cpu资源先让给其他进程<br/>        }        // 如果未获取到锁,且未超时,则继续获取锁<br/>    } while((!$canWrite) && ((microtime() - $startTime) < 1000));    if($canWrite) {        fwrite($fp, $dataToSave);<br/>        flock($file,LOCK_UN);  <br/>    }    fclose($fp);<br/>}<br/></span>

옵션 3: 임시 파일 사용

<span style="font-size: 14px;">$dir_fileopen=&#39;tmp&#39;;function cfopen($filename,$mode){<br/>    global $dir_fileopen;<br/>    clearstatcache();    // 创建一个临时文件<br/>    do{        $id=uniqid();        $tempfilename=$dir_fileopen.&#39;/&#39;.$id.md5($filename);<br/>    } while(file_exists($tempfilename));    // 将要操作的文件内容拷贝到临时文件中<br/>    copy($filename,$tempfilename);    $fp = fopen($tempfilename, $mode);    return $fp ? [$fp, $filename, $id, @filemtime($filename)] : false;<br/>}function cfwrite($fp,$string){<br/>    // 将新增内容写入到临时文件中<br/>    return fwrite($fp[0], $string);<br/>}function cfclose($fp){<br/>    global $dir_fileopen;    $success = fclose($fp[0]);<br/>    clearstatcache();    $tempfilename = $dir_fileopen.&#39;/&#39;.$fp[2].md5($fp[1]);    // 如果要操作的文件在操作期间没有被修改过,则说明没有人操作过该文件,那么将临时文件改名为真正的文件<br/>    if((@filemtime($fp[1]) == $fp[3])){<br/>        rename($tempfilename,$fp[1]);<br/>    }else{        //说明有其它进程在操作目标文件,当前进程被拒绝,删除临时文件<br/>        unlink($tempfilename);        $success = false;<br/>    }    return $success;<br/>}$startTime = microtime();do{    $fp=cfopen(&#39;lock.txt&#39;,&#39;a+&#39;);<br/>    cfwrite($fp,"welcome to beijing.\n");    $success = cfclose($fp, &#39;on&#39;);    if(!$success) {<br/>        usleep(round(rand(0, 100) * 1000));   // 释放cpu,将cpu资源先让给其他进程<br/>    }<br/>}while(!$success && ((microtime() - $startTime) < 1000));    // 如果为false,说明操作失败,则重新进行一次操作<br/></span>

옵션 4: 대기열 사용

파일 작업을 위한 대기열을 생성한 후 쓰기 a 스크립트는 대기열에서 파일 작업 정보를 순차적으로 읽고 파일에 대해 해당 작업을 수행합니다. 이런 방식으로 한 번에 하나의 프로세스만 파일을 작업하므로 동시성 문제가 해결됩니다.

관련 권장 사항:

php

에서 파일 동시 읽기 및 쓰기 충돌 문제 해결

위 내용은 PHP에서 파일을 동시에 읽고 쓰는 문제를 해결하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.