PHP 文件鎖與進(jìn)程鎖的使用示例
鑒于前面介紹了swoole,就借用swoole的服務(wù)器/客戶端與多進(jìn)程機(jī)制對(duì)鎖進(jìn)行說(shuō)明.
這里只針對(duì)PHP的鎖機(jī)制進(jìn)行說(shuō)明,由于SQL的鎖與其作用方式和應(yīng)用場(chǎng)景不同,將作另行說(shuō)明.
1.文件鎖
- flock()
- fclose()
- swoole_lock()
文件鎖的可能應(yīng)用場(chǎng)景為:
1.限制并發(fā)多進(jìn)程或多臺(tái)服務(wù)器需要對(duì)同一文件進(jìn)行訪問(wèn)和修改;
2.對(duì)參與文件I/O的進(jìn)程隊(duì)列化和人為阻塞;
3.在業(yè)務(wù)邏輯中對(duì)文件內(nèi)容進(jìn)行守護(hù);
下面是文件鎖C/S通訊機(jī)制下的使用,已經(jīng)省略了具體的通訊過(guò)程
Server(服務(wù)器通訊過(guò)程已略):
//監(jiān)聽(tīng)數(shù)據(jù)發(fā)送事件
$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ù)器通訊過(guò)程已略):
$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(服務(wù)器通訊過(guò)程已略):
$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();
結(jié)果:
Client2被阻塞了30s,直到Client1執(zhí)行結(jié)束才對(duì)文件進(jìn)行了一次寫(xiě)入;
[l0.16@4 m29.5% c30s04] $ php swoole_client2.php
需要注意的是:
1.無(wú)論是flock()還是swoole提供的swoole_lock(),都有在進(jìn)程結(jié)束時(shí)自動(dòng)解鎖的機(jī)制,所以在demo中即使不進(jìn)行手動(dòng)解鎖也能正常運(yùn)行,因此這里在第一個(gè)Client中執(zhí)行了sleep()暫停函數(shù)來(lái)觀察文件鎖的效果;
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+');
// 如果臨時(shí)文件被鎖定,這里的flock()將返回false
if (!flock($o_file, LOCK_EX + LOCK_NB)) {
var_dump('Process Locked');
}
else {
// 非阻塞模型必須在flock()中增加LOCK_NB參數(shù)
// 當(dāng)然,這里取消LOCK_NB參數(shù)就是阻塞模型了
flock($o_file, LOCK_EX + LOCK_NB);
var_dump('Process Locking');
// 模擬長(zhǎng)時(shí)間的執(zhí)行操作
sleep(10);
}
2.利用swoole提供的共享內(nèi)存,緩存方法或通信方法在不同的進(jìn)程中傳遞一個(gè)全局變量,進(jìn)程獲取該變量的狀態(tài)后使用判定條件控制邏輯執(zhí)行;
傳遞變量的方法很多,這里只提供一個(gè)思路,就以memcached為例;
阻塞模型demo:
// 初始化memcached
$memcached = new Memcache;
$memcached->connect("localhost", 11211);
// 獲取用來(lái)做狀態(tài)判定的全局變量
$s_flag = $memcached->get("flag");
if (!$s_flag) {
// 這里利用了memcached的過(guò)期時(shí)間作為演示,實(shí)際上業(yè)務(wù)處理完成后銷(xiāo)毀該變量即可
$memcached->set("flag", "locked", 0, 10);
main();
}
else {
// 阻塞模型
while ($s_flag == 'locked') {
var_dump('Process locked, retrying...');
// 設(shè)置重試時(shí)間, 避免過(guò)于頻繁的操作嘗試
sleep(1);
// 更新?tīng)顟B(tài)變量
$s_flag = $memcached->get("flag");
}
// // 非阻塞模型
// if ($s_flag == 'locked') {
// var_dump('Process locked, suspended');
// die();
// }
main();
}
// 模擬業(yè)務(wù)主函數(shù)
function main() {
var_dump('Process Running');
// 業(yè)務(wù)執(zhí)行結(jié)束后回收memcached
// $memcached->delete("flag");
}
這里需要注意的是:
1.memcached的過(guò)期時(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ù)器壓力;
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- PHP讀寫(xiě)文件高并發(fā)處理操作實(shí)例詳解
- php處理?yè)屬?gòu)類(lèi)功能的高并發(fā)請(qǐng)求
- php結(jié)合redis實(shí)現(xiàn)高并發(fā)下的搶購(gòu)、秒殺功能的實(shí)例
- php 根據(jù)url自動(dòng)生成縮略圖并處理高并發(fā)問(wèn)題
- PHP flock 文件鎖詳細(xì)介紹
- PHP程序中的文件鎖、互斥鎖、讀寫(xiě)鎖使用技巧解析
- PHP文件鎖定寫(xiě)入實(shí)例解析
- PHP文件鎖函數(shù)flock()詳細(xì)介紹
- PHP基于文件鎖解決多進(jìn)程同時(shí)讀寫(xiě)一個(gè)文件問(wèn)題示例
- PHP使用文件鎖解決高并發(fā)問(wèn)題示例
相關(guān)文章
php定義數(shù)組和使用示例(php數(shù)組的定義方法)
這篇文章主要介紹了php定義數(shù)組和使用示例(php數(shù)組的定義方法),需要的朋友可以參考下2014-03-03
laravel實(shí)現(xiàn)簡(jiǎn)單用戶權(quán)限的示例代碼
這篇文章主要介紹了laravel實(shí)現(xiàn)簡(jiǎn)單用戶權(quán)限的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-05-05
codeigniter自帶數(shù)據(jù)庫(kù)類(lèi)使用方法說(shuō)明
在 CodeIgniter 中,使用數(shù)據(jù)庫(kù)是非常頻繁的事情。你可以使用框架自帶的數(shù)據(jù)庫(kù)類(lèi),就能便捷地進(jìn)行數(shù)據(jù)庫(kù)操作2014-03-03
YII2框架自定義全局函數(shù)的實(shí)現(xiàn)方法小結(jié)
這篇文章主要介紹了YII2框架自定義全局函數(shù)的實(shí)現(xiàn)方法,總結(jié)分析了YII2框架自定義全局函數(shù)相關(guān)實(shí)現(xiàn)技巧與操作注意事項(xiàng),需要的朋友可以參考下2020-03-03
ThinkPHP無(wú)限級(jí)分類(lèi)原理實(shí)現(xiàn)留言與回復(fù)功能實(shí)例
這篇文章主要介紹了ThinkPHP無(wú)限級(jí)分類(lèi)原理實(shí)現(xiàn)留言與回復(fù)功能實(shí)例,并附帶有完整的項(xiàng)目源碼下載供大家學(xué)習(xí)參考,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2014-10-10
PHP 中 Orientation 屬性判斷上傳圖片是否需要旋轉(zhuǎn)
本文給大家介紹使用php技術(shù)實(shí)現(xiàn)根據(jù)上傳圖片orientation屬性判斷是否需要旋轉(zhuǎn),感興趣的朋友一起看看吧2015-10-10

