PHP 實現檔案鎖與進程鎖

來源:互聯網
上載者:User
本篇文章主要介紹了PHP 檔案鎖與進程鎖的使用樣本,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

1.檔案鎖

  • flock()

  • fclose()

  • swoole_lock()

檔案鎖的可能應用情境為:

1.限制並發多進程或多台伺服器需要對同一檔案進行訪問和修改;

2.對參與檔案I/O的進程隊列化和人為阻塞;

3.在商務邏輯中對檔案內容進行守護;

下面是檔案鎖C/S通訊機制下的使用,已經省略了具體的通訊過程

Server(伺服器通訊過程已略):

//監聽資料發送事件$serv->on('receive', function ($serv, $fd, $from_id, $data) {  $serv->send($fd, "ServerEnd");  $p_file = "locktest.txt";  var_dump(file_get_contents($p_file));});

Client1(伺服器通訊過程已略):

$s_recv = "ww";$p_file = "locktest.txt";$o_file = fopen($p_file,'w+');// flock()加鎖方式:flock($o_file,LOCK_EX);// // swoole加鎖方式:// $lock = new swoole_lock(SWOOLE_FILELOCK, $p_file);// $lock->lock();fwrite($o_file, 'ss' . $s_recv);sleep(30);// 兩種解鎖方式// flock($o_file, LOCK_UN);// $lock->unlock();

Client2(伺服器通訊過程已略):

$s_recv = "xx";$p_file = "locktest.txt";$o_file = fopen($p_file,'w+');// flock()加鎖方式:flock($o_file,LOCK_EX);// // swoole加鎖方式:// $lock = new swoole_lock(SWOOLE_FILELOCK, $p_file);// $lock->lock();fwrite($o_file, 'ss' . $s_recv);// 兩種解鎖方式// flock($o_file, LOCK_UN);// $lock->unlock();

結果:

Client2被阻塞了30s,直到Client1執行結束才對檔案進行了一次寫入;

[l0.16@4 m29.5% c30s04] $ php swoole_client2.php

需要注意的是:

1.無論是flock()還是swoole提供的swoole_lock(),都有在進程結束時自動解鎖的機制,所以在demo中即使不進行手動解鎖也能正常運行,因此這裡在第一個Client中執行了sleep()暫停函數來觀察檔案鎖的效果;

2.flock()的標準釋放方式為flock($file,LOCK_UN);, 但是個人喜歡fclose(),永絕後患;

2.進程鎖

與檔案鎖不同的是,進程鎖並不用於阻止對檔案的I/O,而是用於防止多進程並發造成的預期之外的後果.所以需要在多進程並發時將其隊列化,即在某進程的關鍵邏輯執行結束前阻塞其他並發進程的邏輯執行.

實現思路有幾種:

1.利用flock()檔案鎖,建立一個臨時lock檔案,使用LOCK_NB類比阻塞或非阻塞流,再在進程內部使用判定條件控制邏輯執行;

非阻塞模型demo:

$p_file = "locktest.txt";$o_file = fopen($p_file, 'w+');// 如果臨時檔案被鎖定,這裡的flock()將返回falseif (!flock($o_file, LOCK_EX + LOCK_NB)) {  var_dump('Process Locked');}else {  // 非阻塞模型必須在flock()中增加LOCK_NB參數  // 當然,這裡取消LOCK_NB參數就是阻塞模型了  flock($o_file, LOCK_EX + LOCK_NB);  var_dump('Process Locking');  // 類比長時間的執行操作  sleep(10);}

2.利用swoole提供的共用記憶體,緩衝方法或通訊方法在不同的進程中傳遞一個全域變數,進程擷取該變數的狀態後使用判定條件控制邏輯執行;

傳遞變數的方法很多,這裡只提供一個思路,就以memcached為例;

阻塞模型demo:

// 初始化memcached$memcached = new Memcache;$memcached->connect("localhost", 11211);// 擷取用來做狀態判定的全域變數$s_flag = $memcached->get("flag");if (!$s_flag) {  // 這裡利用了memcached的到期時間作為示範,實際上業務處理完成後銷毀該變數即可  $memcached->set("flag", "locked", 0, 10);  main();}else {  // 阻塞模型  while ($s_flag == 'locked') {    var_dump('Process locked, retrying...');    // 設定重試時間, 避免過於頻繁的操作嘗試    sleep(1);    // 更新狀態變數    $s_flag = $memcached->get("flag");  }  // // 非阻塞模型  // if ($s_flag == 'locked') {  //   var_dump('Process locked, suspended');  //   die();  // }  main();}// 類比業務主函數function main() {  var_dump('Process Running');  // 業務執行結束後回收memcached  // $memcached->delete("flag");}

這裡需要注意的是:

1.memcached的到期時間不可少於程式啟動並執行實際時間,因此建議稍微長一點,邏輯執行結束後進行回收;

2.在非阻塞模型中,若狀態被判定為false,應該將進程中止或block,避免商務邏輯的繼續執行;

3.在實際應用中,設定一個重試時間很有必要,這樣可以很大程度上減少針對memcached的大量I/O並發,減輕伺服器壓力;


相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.