亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > PHP > 正文

php中并發讀寫文件沖突的解決方案

2024-05-04 23:18:57
字體:
來源:轉載
供稿:網友

對于日IP不高或者說并發數不是很大的應用,一般不用考慮這些!用一般的文件操作方法完全沒有問題。但如果并發高,在我們對文件進行讀寫操作時,很有可能多個進程對進一文件進行操作,如果這時不對文件的訪問進行相應的獨占,就容易造成數據丟失。
例如:一個在線聊天室(這里假定把聊天內容寫入文件),在同一時刻,用戶A和用戶B都要操作數據保存文件,首先是A打開了文件,然后更新里面的數據,但這里B也正好也打開了同一個文件,也準備更新里面的數據。當A把寫好的文件保存時,這里其實B已經打開了文件。但當B再把文件保存回去時,這里已經造成了數據的丟失,因為這里B用戶完全不知道它所打開的文件在它對其進行更改時,A用戶也更改了這個文件,所以最后B用戶保存更改時,用戶A的更新就被會丟失。
對于這樣的問題,一般的解決方案時當一進程對文件進行操作時,首先對其它進行加鎖,意味著這里只有該進程有權對文件進行讀取,其它進程如果現在讀,是完全沒有問題,但如果這時有進程試圖想對其進行更新,會遭到操作拒絕,先前對文件進行加鎖的進程這時如果對文件的更新操作完畢,這就釋放獨占的標識,這時文件又恢復到了可更改的狀態。接下來同理,如果那個進程在操作文件時,文件沒有加鎖,這時,它就可以放心大膽的對文件進行鎖定,獨自享用。
一般的方案會是:

復制代碼 代碼如下:


$fp=fopen('/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);


但在PHP中,flock似乎工作的不是那么好!在多并發情況下,似乎是經常獨占資源,不即時釋放,或者是根本不釋放,造成死鎖,從而使服務器的cpu占用很高,甚至有時候會讓服務器徹底死掉。好像在很多linux/unix系統中,都會有這樣的情況發生。所以使用flock之前,一定要慎重考慮。
那么就沒有解決方案了嗎?其實也不是這樣的。如果flock()我們使用得當,完全可能解決死鎖的問題。當然如果不考慮使用flock()函數,也同樣會有很好的解決方案來解決我們的問題。經過我個人的搜集和總結,大致歸納了解決方案有如下幾種。
方案一:對文件進行加鎖時,設置一個超時時間。大致實現如下:

復制代碼 代碼如下:


if($fp=fopen($fileName,'a')){
 $startTime=microtime();
 do{
  $canWrite=flock($fp,LOCK_EX);
  if(!$canWrite){
   usleep(round(rand(0,100)*1000));
  }
 }while((!$canWrite)&&((microtime()-$startTime)<1000));
 if($canWrite){
  fwrite($fp,$dataToSave);
 }
 fclose($fp);
}


超時設置為1ms,如果這里時間內沒有獲得鎖,就反復獲得,直接獲得到對文件操作權為止,當然。如果超時限制已到,就必需馬上退出,讓出鎖讓其它進程來進行操作。

方案二:不使用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沒來得及更新緩存里的文件屬性,容易導致訪問到最后更新時間不是真實的數據。所以這里需要使用該函數對已保存的緩存進行清除。

方案三:對操作的文件進行隨機讀寫,以降低并發的可能性。
在對用戶訪問日志進行記錄時,這種方案似乎被采用的比較多。先前需要定義一個隨機空間,空間越大,并發的的可能性就越小,這里假設隨機讀寫空間為[1-500],那么我們的日志文件的分布就為log1~到log500不等。每一次用戶訪問,都將數據隨機寫到log1~log500之間的任一文件。在同一時刻,有2個進程進行記錄日志,A進程可能是更新的log32文件,而B進程呢?則此時更新的可能就為log399.要知道,如果要讓B進程也操作log32,概率基本上為1/500,差不多約等于零。在需要對訪問日志進行分析時,這里我們只需要先將這些日志合并,再進行分析即可。使用這種方案來記錄日志的一個好處時,進程操作排隊的可能性比較小,可以使進程很迅速的完成每一次操作。

方案四:將所有要操作的進程放入一個隊列中。然后專門放一個服務完成文件操作。隊列中的每一個排除的進程相當于第一個具體的操作,所以第一次我們的服務只需要從隊列中取得相當于具體操作事項就可以了,如果這里還有大量的文件操作進程,沒關系,排到我們的隊列后面即可,只要愿意排,隊列的多長都沒關系。

對于以前幾種方案,各有各的好處!大致可能歸納為兩類:
(1)需要排隊(影響慢)比如方案一、二、四
(2)不需要排隊。(影響快)方案三
在設計緩存系統時,一般我們不會采用方案三。因為方案三的分析程序和寫入程序是不同步的,在寫的時間,完全不考慮到時候分析的難度,只管寫的行了。試想一下,如我們在更新一個緩存時,如果也采用隨機文件讀寫法,那么在讀緩存時似乎會增加很多流程。但采取方案一、二就完全不一樣,雖然寫的時間需要等待(當獲取鎖不成功時,會反復獲?。x文件是很方便的。添加緩存的目的就是要減少數據讀取瓶頸,從而提高系統性能。
從上為個人經驗和一些資料的總結,有什么不對的地方,或者沒有談到的地方,歡迎各位同行指正。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久在线免费视频| 欧美精品激情视频| 欧美日韩国产精品专区| 91亚洲精华国产精华| 庆余年2免费日韩剧观看大牛| 成人福利网站在线观看11| 国产精品久久二区| 久久精品国产v日韩v亚洲| 亚洲欧美日韩区| 欧美午夜丰满在线18影院| 亚洲综合在线播放| 亚洲激情第一页| 欧美另类极品videosbestfree| www.日本久久久久com.| 中文字幕av一区中文字幕天堂| 久久久人成影片一区二区三区观看| 亚洲激情视频在线播放| 欧美成年人视频网站| 精品国内自产拍在线观看| 欧美国产乱视频| 亚洲精品国精品久久99热一| 久久九九热免费视频| 成人黄色av免费在线观看| 欧美中文字幕视频在线观看| 91成人精品网站| 国产精品普通话| 亚洲国产成人精品一区二区| 国产女人18毛片水18精品| 粉嫩老牛aⅴ一区二区三区| 亚洲精品狠狠操| 日本高清视频精品| 亚洲a中文字幕| 永久免费精品影视网站| 欧美精品久久久久久久免费观看| 亚洲的天堂在线中文字幕| 久99九色视频在线观看| 欧美整片在线观看| 国产精品27p| 国产一区二区久久精品| 久久黄色av网站| 欧美日韩国产中文精品字幕自在自线| 欧美日韩一区二区三区在线免费观看| 色综合久久悠悠| 久久国产一区二区三区| 日韩欧美中文字幕在线播放| 久久久精品久久| 国内精品小视频| 亚洲成色www8888| 日本精品在线视频| 插插插亚洲综合网| 国产精品高精视频免费| 亚洲欧美日韩国产中文| 激情成人在线视频| 91精品在线看| 91精品在线播放| 午夜精品视频网站| 国产精品视频男人的天堂| 欧美成人第一页| 色综合五月天导航| 欧美视频专区一二在线观看| 91精品国产自产91精品| 中文字幕国产亚洲| 亚洲欧美国产一本综合首页| 欧美一级视频免费在线观看| 国产日韩精品视频| 亚洲视频免费一区| 久久全国免费视频| 中文字幕最新精品| 欧美激情免费在线| 国产一区二区三区免费视频| 国产在线观看91精品一区| 亚洲女人天堂成人av在线| 精品亚洲一区二区三区| 日韩视频欧美视频| 18一19gay欧美视频网站| 亚洲成人网久久久| 一区二区日韩精品| 色婷婷亚洲mv天堂mv在影片| 亚洲欧美国产一区二区三区| 久久久国产精品x99av| 精品国产乱码久久久久酒店| 欧美在线观看www| 亚洲欧美在线看| 欧美一级bbbbb性bbbb喷潮片| 午夜欧美大片免费观看| 91爱爱小视频k| 在线视频中文亚洲| www.久久草.com| 91免费精品国偷自产在线| 国产精品自产拍在线观| 亚洲影视九九影院在线观看| 国内精品久久久久伊人av| 97久久精品人搡人人玩| 国产极品jizzhd欧美| 国产精品免费久久久久影院| 97国产在线观看| 欧美亚洲成人xxx| 亚洲xxxx视频| 久久久久久久久久亚洲| 日韩成人在线视频观看| 韩国一区二区电影| 久久久久久久久久久免费| 精品亚洲一区二区三区四区五区| 国产91对白在线播放| 这里只有精品视频| 欧美高清在线观看| 亚洲天堂免费视频| 成人情趣片在线观看免费| 日韩免费观看网站| 欧美精品亚州精品| 亚洲精品电影在线| 亚洲欧美日韩精品久久奇米色影视| 欧美日韩国内自拍| 国内自拍欧美激情| 91精品国产自产91精品| 亚洲日本aⅴ片在线观看香蕉| 亚洲欧美一区二区激情| 国产91av在线| 国产亚洲综合久久| 久久久免费精品| 亚洲免费小视频| 久久综合久中文字幕青草| 亚洲欧美国内爽妇网| 国产精品69久久久久| 日韩在线观看电影| 久久久久久国产精品久久| 欧美激情日韩图片| 中文字幕日韩高清| 欧美另类在线播放| 色偷偷av一区二区三区乱| 人人做人人澡人人爽欧美| 欧美一级视频免费在线观看| 欧美巨猛xxxx猛交黑人97人| 久久久久久久久综合| 国内精品伊人久久| 欧美日韩中文字幕| 久久久久国产精品免费网站| 欧美视频免费在线| 久久久国产视频91| 国产性猛交xxxx免费看久久| 亚洲国产精品电影在线观看| 国产综合福利在线| 波霸ol色综合久久| 日韩av三级在线观看| 91精品国产色综合久久不卡98| 欧美中文字幕精品| 亚洲欧洲国产一区| 国产视频自拍一区| 26uuu亚洲伊人春色| 国产精品久久久久国产a级| 欧美综合在线第二页| 国产精品h片在线播放| 亚洲国产精品久久精品怡红院| 亚洲欧美一区二区精品久久久| 91精品一区二区| 欧美贵妇videos办公室| 91探花福利精品国产自产在线| 久久视频在线直播| 亚洲精品国产电影| 日韩av片永久免费网站| 久久久久九九九九| 91老司机在线| 欧美性极品少妇精品网站| 亚洲91av视频|