php怎麼解決多進程同時寫一個檔案的問題?

2020-07-16 10:06:27

php解決多進程同時寫一個檔案的方法:首先複製需要更新的檔案,並更改檔名;然後檢查最後更新時間和先前所儲存的時間是否一致;最後將所修改的臨時檔案重新命名到原檔案。


php解決多進程同時寫一個檔案的方法:

首先PHP是支援進程的而不支援多執行緒(這個先搞清楚了),如果是對於檔案操作,其實你只需要給檔案加鎖就能解決,不需要其它操作,PHP的flock已經幫你搞定了。
用flock在寫檔案前先鎖上,等寫完後解鎖,這樣就實現了多執行緒同時讀寫一個檔案避免衝突。大概就是下面這個流程

/*
*flock(file,lock,block)
*file 必需,規定要鎖定或釋放的已開啟的檔案
*lock 必需。規定要使用哪種鎖定型別。
*block 可選。若設定為 1 或 true,則當進行鎖定時阻擋其他進程。
*lock
*LOCK_SH 要取得共用鎖定(讀取的程式)
*LOCK_EX 要取得獨占鎖定(寫入的程式)
*LOCK_UN 要釋放鎖定(無論共用或獨占)
*LOCK_NB 如果不希望 flock() 在鎖定時堵塞
/*
if (flock($file,LOCK_EX))
{
fwrite($file,'write more words');
flock($file,LOCK_UN);
}
else
{
//處理錯誤邏輯
}
fclose($file);
)

方案:不使用flock函數,借用臨時檔案來解決讀寫衝突的問題。

大致原理如下:

(1)將需要更新的檔案考慮一份到我們的臨時檔案目錄,將檔案最後修改時間儲存到一個變數,並為這個臨時檔案取一個隨機的,不容易重複的檔名。

(2)當對這個臨時檔案進行更新後,再檢測原檔案的最後更新時間和先前所儲存的時間是否一致。

(3)如果最後一次修改時間一致,就將所修改的臨時檔案重新命名到原檔案,為了確保檔案狀態同步更新,所以需要清除一下檔案狀態。

(4)但是,如果最後一次修改時間和先前所儲存的一致,這說明在這期間,原檔案已經被修改過,這時,需要把臨時檔案刪除,然後返回false,說明檔案這時有其它進程在進行操作。

實現程式碼如下:

程式碼如下:

$dir_fileopen='tmp';
function randomid(){
    return time().substr(md5(microtime()),0,rand(5,12));
}
function cfopen($filename,$mode){
    global $dir_fileopen;
    clearstatcache();
    do{
  $id=md5(randomid(rand(),TRUE));
        $tempfilename=$dir_fileopen.'/'.$id.md5($filename);
    } while(file_exists($tempfilename));
    if(file_exists($filename)){
        $newfile=false;
        copy($filename,$tempfilename);
    }else{
        $newfile=true;
    }
    $fp=fopen($tempfilename,$mode);
    return $fp?array($fp,$filename,$id,@filemtime($filename)):false;
}
function cfwrite($fp,$string){
 return fwrite($fp[0],$string);
}
function cfclose($fp,$debug='off'){
    global $dir_fileopen;
    $success=fclose($fp[0]);
    clearstatcache();
    $tempfilename=$dir_fileopen.'/'.$fp[2].md5($fp[1]);
    if((@filemtime($fp[1])==$fp[3])||($fp[4]==true&&!file_exists($fp[1]))||$fp[5]==true){
        rename($tempfilename,$fp[1]);
    }else{
        unlink($tempfilename);
  //說明有其它進程 在操作目標檔案,當前進程被拒絕
        $success=false;
    }
    return $success;
}
$fp=cfopen('lock.txt','a+');
cfwrite($fp,"welcome to beijing.n");
fclose($fp,'on');

對於上面的程式碼所使用的函數,需要說明一下:

(1)rename();重新命名一個檔案或一個目錄,該函數其實更像linux裡的mv。更新檔案或者目錄的路徑或名字很方便。但當我在window測試上面程式碼時,如果新檔名已經存在,會給出一個notice,說當前檔案已經存在。但在linux下工作的很好。

(2)clearstatcache();清除檔案的狀態.php將快取所有檔案屬性資訊,以提供更高的效能,但有時,多進程在對檔案進行刪除或者更新操作時,php沒來得及更新快取裡的檔案屬性,容易導致存取到最後更新時間不是真實的資料。所以這裡需要使用該函數對已儲存的快取進行清除。

以上就是php怎麼解決多進程同時寫一個檔案的問題?的詳細內容,更多請關注TW511.COM其它相關文章!