hz_chenwenbiao php文件鎖(轉)

bool flock ( int handle, int operation [, int &wouldblock] );
flock() 操做的 handle 必須是一個已經打開的文件指針。operation 能夠是如下值之一:php

  1. 要取得共享鎖定(讀取程序),將 operation 設爲 LOCK_SH(PHP 4.0.1 之前的版本設置爲 1)
  2. 要取得獨佔鎖定(寫入程序),將 operation 設爲 LOCK_EX(PHP 4.0.1 之前的版本中設置爲 2)
  3. 要釋放鎖定(不管共享或獨佔),將 operation 設爲 LOCK_UN(PHP 4.0.1 之前的版本中設置爲 3)
  4. 若是你不但願 flock() 在鎖定時堵塞,則給 operation 加上 LOCK_NB(PHP 4.0.1 之前的版本中設置爲 4)

建兩個文件
(1) a.phphtml

?
$file = "temp.txt" ;   
$fp = fopen ( $file , 'w' );   
if ( flock ( $fp , LOCK_EX)){   
      fwrite( $fp , "abc\n" );   
      sleep(10);   
      fwrite( $fp , "123\n" );   
     flock ( $fp , LOCK_UN);   
}   
fclose( $fp );  

(2) b.php數據庫

?
$file = "temp.txt" ;   
$fp = fopen ( $file , 'r' );   
echo fread ( $fp , 100);   
fclose( $fp );  

運行 a.php 後,立刻運行 b.php ,能夠看到輸出:
abc
等 a.php 運行完後運行 b.php ,能夠看到輸出:
abc
123
顯然,當 a.php 寫文件時數據太大,致使時間比較長時,這時 b.php 讀取數據不完整緩存

修改 b.php 爲:測試

?
$file = "temp.txt" ;   
$fp = fopen ( $file , 'r' );   
if ( flock ( $fp , LOCK_EX)){   
     echo fread ( $fp , 100);   
     flock ( $fp , LOCK_UN);   
} else {   
     echo "Lock file failed...\n" ;   
}   
fclose( $fp );  

運行 a.php 後,立刻運行 b.php ,能夠發現 b.php 會等到 a.php 運行完成後(即 10 秒後)才顯示:
abc
123
讀取數據完整,但時間過長,他要等待寫鎖釋放。spa

修改 b.php 爲:指針

?
$file = "temp.txt" ;   
$fp = fopen ( $file , 'r' );   
if ( flock ( $fp , LOCK_SH | LOCK_NB)){   
     echo fread ( $fp , 100);   
     flock ( $fp , LOCK_UN);   
} else {   
     echo "Lock file failed...\n" ;   
}   
fclose( $fp );  

運行 a.php 後,立刻運行 b.php ,能夠看到輸出:
Lock file failed…
證實能夠返回鎖文件失敗狀態,而不是向上面同樣要等好久。code

結論:
建議做文件緩存時,選好相關的鎖,否則可能致使讀取數據不完整,或重複寫入數據。
file_get_contents 好像選擇不了鎖,不知道他默認用的什麼鎖,反正和不鎖獲得的輸出同樣,是不完整的數據。
我是要作文件緩存,因此只須要知道是否有寫鎖存在便可,有的話就查數據庫就能夠了。
測試環境:Linux(Ubuntu 6) , PHP 5.1.2 , Apache 2htm

再轉:blog

文件鎖有兩種:共享鎖和排他鎖,也就是讀鎖(LOCK_SH)和寫鎖(LOCK_EX) 
文件的鎖通常這麼使用:
  1. $fp = fopen("filename""a");   
  2. flock($fp, LOCK_SH) or die("lock error")   
  3. $str = fread($fp, 1024);   
  4. flock($fp, LOCK_UN);   
  5. fclose($fp);  

注意fwrite以後,文件當即就被更新了,而不是等fwrite而後fclose以後文件纔會更新,這個能夠經過在fwrite以後fclose以前讀取這個文件進行檢查 

可是何時使用lock_ex何時使用lock_sh呢? 

讀的時候: 
若是不想出現dirty數據,那麼最好使用lock_sh共享鎖。能夠考慮如下三種狀況: 
1. 若是讀的時候沒有加共享鎖,那麼其餘程序要寫的話(無論這個寫是加鎖仍是不加鎖)都會當即寫成功。若是正好讀了一半,而後被其餘程序給寫了,那麼讀的後一半就有可能跟前一半對不上(前一半是修改前的,後一半是修改後的) 
2. 若是讀的時候加上了共享鎖(由於只是讀,沒有必要使用排他鎖),這個時候,其餘程序開始寫,這個寫程序沒有使用鎖,那麼寫程序會直接修改這個文件,也會致使前面同樣的問題 
3. 最理想的狀況是,讀的時候加鎖(lock_sh),寫的時候也進行加鎖(lock_ex),這樣寫程序會等着讀程序完成以後才進行操做,而不會出現貿然操做的狀況 

寫的時候: 
若是多個寫程序不加鎖同時對文件進行操做,那麼最後的數據有可能一部分是a程序寫的,一部分是b程序寫的 
若是寫的時候加鎖了,這個時候有其餘的程序來讀,那麼他會讀到什麼東西呢? 
1. 若是讀程序沒有申請共享鎖,那麼他會讀到dirty的數據。好比寫程序要寫a,b,c三部分,寫完a,這時候讀讀到的是a,繼續寫b,這時候讀讀到的是ab,而後寫c,這時候讀到的是abc. 
2. 若是讀程序在以前申請了共享鎖,那麼讀程序會等寫程序將abc寫完並釋放鎖以後才進行讀。 

還有一篇也寫得不錯的博文:

http://hxsdit.com/1110
相關文章
相關標籤/搜索