亚洲国产日韩欧美在线a乱码,国产精品路线1路线2路线,亚洲视频一区,精品国产自,www狠狠,国产情侣激情在线视频免费看,亚洲成年网站在线观看

PHP文件鎖與進(jìn)程鎖

時(shí)間:2025-11-02 19:35:22 php語言

PHP文件鎖與進(jìn)程鎖

  PHP具有非常強(qiáng)大的功能,所有的CGI的功能PHP都能實(shí)現(xiàn)。下面是小編分享的PHP文件鎖與進(jìn)程鎖,一起來看一下吧。

  借用swoole的服務(wù)器/客戶端與多進(jìn)程機(jī)制對(duì)鎖進(jìn)行說明.

  這里只針對(duì)PHP的鎖機(jī)制進(jìn)行說明,由于SQL的鎖與其作用方式和應(yīng)用場(chǎng)景不同,將作另行說明.

  1.文件鎖

  flock()

  fclose()

  swoole_lock()

  文件鎖的可能應(yīng)用場(chǎng)景為:

  1.限制并發(fā)多進(jìn)程或多臺(tái)服務(wù)器需要對(duì)同一文件進(jìn)行訪問和修改;

  2.對(duì)參與文件I/O的進(jìn)程隊(duì)列化和人為阻塞;

  3.在業(yè)務(wù)邏輯中對(duì)文件內(nèi)容進(jìn)行守護(hù);

  下面是文件鎖C/S通訊機(jī)制下的使用,已經(jīng)省略了具體的通訊過程,如有需要請(qǐng)移步swoole異步任務(wù)隊(duì)列

  Server(服務(wù)器通訊過程已略):

  /pic/p>

  $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(服務(wù)器通訊過程已略):

  $s_recv = "ww";

  $p_file = "locktest.txt";

  $o_file = fopen($p_file,'w+');/pic/p>

  flock($o_file,LOCK_EX);/pic/pic/p>

  /pic/p>

  /pic/p>

  fwrite($o_file, 'ss' . $s_recv);

  sleep(30);/pic/p>

  /pic/p>

  /pic/p>

  Client2(服務(wù)器通訊過程已略):

  $s_recv = "xx";

  $p_file = "locktest.txt";

  $o_file = fopen($p_file,'w+');/pic/p>

  flock($o_file,LOCK_EX);/pic/p>

  /pic/p>

  /pic/p>

  /pic/p>

  fwrite($o_file, 'ss' . $s_recv);/pic/p>

  /pic/p>

  /pic/p>

  結(jié)果:

  Client2被阻塞了30s,直到Client1執(zhí)行結(jié)束才對(duì)文件進(jìn)行了一次寫入;

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

  需要注意的是:

  1.無論是flock()還是swoole提供的swoole_lock(),都有在進(jìn)程結(jié)束時(shí)自動(dòng)解鎖的機(jī)制,所以在demo中即使不進(jìn)行手動(dòng)解鎖也能正常運(yùn)行,因此這里在第一個(gè)Client中執(zhí)行了sleep()暫停函數(shù)來觀察文件鎖的效果;

  2.flock()的標(biāo)準(zhǔn)釋放方式為flock($file,LOCK_UN);, 但是個(gè)人喜歡fclose(),永絕后患;

  2.進(jìn)程鎖

  與文件鎖不同的是,進(jìn)程鎖并不用于阻止對(duì)文件的I/O,而是用于防止多進(jìn)程并發(fā)造成的預(yù)期之外的后果.所以需要在多進(jìn)程并發(fā)時(shí)將其隊(duì)列化,即在某進(jìn)程的關(guān)鍵邏輯執(zhí)行結(jié)束前阻塞其他并發(fā)進(jìn)程的邏輯執(zhí)行.

  實(shí)現(xiàn)思路有幾種:

  1.利用flock()文件鎖,創(chuàng)建一個(gè)臨時(shí)lock文件,使用LOCK_NB模擬阻塞或非阻塞流,再在進(jìn)程內(nèi)部使用判定條件控制邏輯執(zhí)行;

  非阻塞模型demo:

  $p_file = "locktest.txt";

  $o_file = fopen($p_file, 'w+');/pic/p>

  if (!flock($o_file, LOCK_EX + LOCK_NB)) {

  var_dump('Process Locked');

  }else {    /pic/p>

  /pic/p>

  flock($o_file, LOCK_EX + LOCK_NB);

  var_dump('Process Locking');    /pic/p>

  sleep(10);

  }

  2.利用swoole提供的共享內(nèi)存,緩存方法或通信方法在不同的進(jìn)程中傳遞一個(gè)全局變量,進(jìn)程獲取該變量的狀態(tài)后使用判定條件控制邏輯執(zhí)行;

  傳遞變量的方法很多,這里只提供一個(gè)思路,就以memcached為例;

  阻塞模型demo:

  /pic/p>

  $memcached->connect("localhost", 11211);/pic/p>

  $s_flag = $memcached->get("flag");

  if (!$s_flag) {

  /pic/p>

  $memcached->set("flag", "locked", 0, 10);

  main();

  }else {

  /pic/p>

  while ($s_flag == 'locked') {

  var_dump('Process locked, retrying...');

  /pic/p>

  sleep(1);        /pic/p>

  $s_flag = $memcached->get("flag");

  }

  /pic/pic/p>

  /pic/p>

  /pic/p>

  /pic/p>

  /pic/p>

  main();

  }/pic/p>

  var_dump('Process Running');  /pic/p>

  /pic/p>

  }

  這里需要注意的是:

  1.memcached的過期時(shí)間不可少于程序運(yùn)行的實(shí)際時(shí)間,因此建議稍微長(zhǎng)一點(diǎn),邏輯執(zhí)行結(jié)束后進(jìn)行回收;

  2.在非阻塞模型中,若狀態(tài)被判定為false,應(yīng)該將進(jìn)程中止或block,避免業(yè)務(wù)邏輯的繼續(xù)執(zhí)行;

  3.在實(shí)際應(yīng)用中,設(shè)置一個(gè)重試時(shí)間很有必要,這樣可以很大程度上減少針對(duì)memcached的大量I/O并發(fā),減輕服務(wù)器壓力;


【PHP文件鎖與進(jìn)程鎖】相關(guān)文章:

PHP文件鎖與進(jìn)程鎖的實(shí)現(xiàn)12-12

php文件鎖怎么用11-17

linux新手教程之創(chuàng)建鎖文件的方法08-22

金鎖銀鎖大班教案01-26

托班的游戲教案:金鎖銀鎖02-15

如何打開php文件 php文件怎么打開01-13

PHP文件是什么 如何打開PHP文件03-06

認(rèn)識(shí)鎖的教案參考03-08

Fireworks如何制作屬鎖11-01