成人精品一区二区三区中文字幕-成人精品一区二区三区-成人精品一级毛片-成人精品亚洲-日本在线视频一区二区-日本在线视频免费

導航首頁 ? 技術教程 ? PHP 文件鎖與進程鎖的使用示例
全站頭部文字 我要出現在這里
PHP 文件鎖與進程鎖的使用示例 763 2023-12-10   

鑒于前面介紹了swoole,就借用swoole的服務器/客戶端與多進程機制對鎖進行說明.

這里只針對PHP的鎖機制進行說明,由于SQL的鎖與其作用方式和應用場景不同,將作另行說明.

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()將返回false
if (!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并發,減輕服務器壓力;

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持綠夏網。


主站蜘蛛池模板: 张小斐和雷佳音的电影| 男女高清视频| 松雪泰子| 电影《48天》免费观看全集| 小丑回魂1| 在线电影免费| midjourney中文版| 抖色| 漂亮主妇| 美丽女波士1983| 寄宿生韩剧全集观看| 从此以后歌词| 裸舞在线观看| 抖音app安装| 刘浩存《一秒钟》舞蹈| 太原教育电视台| 真实游戏完整版高清观看| 贝的故事教案设计优秀教案| 铠甲勇士雷霆雅塔莱斯| 李泽锋演过的所有电视剧| 又见阿郎电视剧免费观看| 艳肉观世音性三级| 魔法少女砂沙美| 颂赞诗歌| 好好操视频| 知否知否应是绿肥红瘦电视剧免费| 横冲直撞好莱坞| 黑之教室| 色女孩视频| 掩护| 情欲狂欢电影| 澳大利亚《囚犯》| 燃冬海报| 闪电11人| 浙江卫视回放观看入口| 爱情与灵药 电影| 中国宇航员遇难| 常乐镇| 公共基础知识1000题及答案| 星河长明免费观看电视剧| 十一个月宝宝发育标准|

!!!站長長期在線接!!!

網站、小程序:定制開發/二次開發/仿制開發等

各種疑難雜癥解決/定制接口/定制采集等

站長微信:lxwl520520

站長QQ:1737366103