日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

PHP編程中的鎖

瀏覽:262日期:2022-09-12 13:11:56

最近看了《理解Linux進程》這本開源書,鏈接。該書描述了linux中的進程概念,對鎖和進程間通信(IPC)有一些總結。不過該書的描述語言是golang, 平時用的比較少,就想對應概念找找php中的接口。

文件鎖

全名叫advisory file lock, 書中有提及。 這類鎖比較常見,例如 mysql, php-fpm 啟動之后都會有一個pid文件記錄了進程id,這個文件就是文件鎖。

這個鎖可以防止重復運行一個進程,例如在使用crontab時,限定每一分鐘執行一個任務,但這個進程運行時間可能超過一分鐘,如果不用進程鎖解決沖突的話兩個進程一起執行就會有問題。

使用PID文件鎖還有一個好處,方便進程向自己發停止或者重啟信號。例如重啟php-fpm的命令為

kill -USR2 `cat /usr/local/php/var/run/php-fpm.pid`

發送USR2信號給pid文件記錄的進程,信號屬于進程通信,會另開一個篇幅。

php的接口為flock,文檔比較詳細。先看一下定義,bool flock ( resource $handle , int $operation [, int &$wouldblock ] ).

$handle是文件系統指針,是典型地由 fopen() 創建的 resource(資源)。這就意味著使用flock必須打開一個文件。

$operation是操作類型。

&$wouldblock如果鎖是阻塞的,那么這個變量會設為1.

需要注意的是,這個函數默認是阻塞的,如果想非阻塞可以在 operation 加一個 bitmaskLOCK_NB. 接下來測試一下。

$pid_file = '/tmp/process.pid';$pid = posix_getpid();$fp = fopen($pid_file, ’w+’);if(flock($fp, LOCK_EX | LOCK_NB)){ echo 'got the lock n'; ftruncate($fp, 0); // truncate file fwrite($fp, $pid); fflush($fp); // flush output before releasing the lock sleep(300); // long running process flock($fp, LOCK_UN); // 釋放鎖定} else { echo 'Cannot get pid lock. The process is already up n';}fclose($fp);

保存為process.php,運行php process.php &, 此時再次運行php process.php,就可以看到錯誤提示。flock也有共享鎖,LOCK_SH.

互斥鎖和讀寫鎖sync模塊中的Mutex

Mutex是一個組合詞,mutual exclusion。用pecl安裝一下sync模塊,pecl install sync。 文檔中的SyncMutex只有兩個方法,lock 和 unlock, 我們就直接上代碼測試吧。沒有用IDE寫,所以cs異常丑陋,請無視。

$mutex = new SyncMutex('UniqueName');for($i=0; $i<2; $i++){ $pid = pcntl_fork(); if($pid <0){die('fork failed'); }elseif ($pid>0){echo 'parent process n'; }else{echo 'child process {$i} is born. n';obtainLock($mutex, $i); }}while (pcntl_waitpid(0, $status) != -1) { $status = pcntl_wexitstatus($status); echo 'Child $status completedn'; }function obtainLock ($mutex, $i){ echo 'process {$i} is getting the mutex n'; $res = $mutex->lock(200); sleep(1); if (!$res){echo 'process {$i} unable to lock mutex. n'; }else{echo 'process {$i} successfully got the mutex n';$mutex->unlock(); } exit();}

保存為mutex.php, runphp mutex.php, output is

parent process parent process child process 1 is born. process 1 is getting the mutex child process 0 is born. process 0 is getting the mutex process 1 successfully got the mutex Child 0 completedprocess 0 unable to lock mutex. Child 0 completed

這里子進程0和1不一定誰在前面。但是總有一個得不到鎖。這里SyncMutex::lock(int $millisecond)的參數是 millisecond, 代表阻塞的時長, -1 為無限阻塞。

sync模塊中的讀寫鎖

SyncReaderWriter的方法類似,readlock,readunlock,writelock,writeunlock,成對出現即可,沒有寫測試代碼,應該和Mutex的代碼一致,把鎖替換一下就可以。

sync模塊中的Event

感覺和golang中的Cond比較像,wait()阻塞,fire()喚醒Event阻塞的一個進程。有一篇好文介紹了Cond, 可以看出Cond就是鎖的一種固定用法。SyncEvent也一樣。php文檔中的例子顯示,fire()方法貌似可以用在web應用中。

上測試代碼

for($i=0; $i<3; $i++){ $pid = pcntl_fork(); if($pid <0){die('fork failed'); }elseif ($pid>0){//echo 'parent process n'; }else{echo 'child process {$i} is born. n';switch ($i) {case 0: wait(); break;case 1: wait(); break;case 2: sleep(1); fire(); break;} }}while (pcntl_waitpid(0, $status) != -1) { $status = pcntl_wexitstatus($status); echo 'Child $status completedn'; }function wait(){ $event = new SyncEvent('UniqueName'); echo 'before waiting. n'; $event->wait(); echo 'after waiting. n'; exit();}function fire(){ $event = new SyncEvent('UniqueName'); $event->fire(); exit();}

這里故意少寫一個fire(), 所以程序會阻塞,證明了 fire() 一次只喚醒一個進程。

pthreads模塊

貌似也看到了Mutex, Cond, Pool. 沒來得及看,看完再補充。

信號量sync模塊中的信號量

SyncSemaphore文檔中顯示,它和Mutex的不同之處,在于Semaphore一次可以被多個進程(或線程)得到,而Mutex一次只能被一個得到。所以在SyncSemaphore的構造函數中,有一個參數指定信號量可以被多少進程得到。public SyncSemaphore::__construct ([ string $name [, integer $initialval [, bool $autounlock ]]] )就是這個$initialval(initial value)

$lock = new SyncSemaphore('UniqueName', 2);for($i=0; $i<2; $i++){ $pid = pcntl_fork(); if($pid <0){die('fork failed'); }elseif ($pid>0){echo 'parent process n'; }else{echo 'child process {$i} is born. n';obtainLock($lock, $i); }}while (pcntl_waitpid(0, $status) != -1) { $status = pcntl_wexitstatus($status); echo 'Child $status completedn'; }function obtainLock ($lock, $i){ echo 'process {$i} is getting the lock n'; $res = $lock->lock(200); sleep(1); if (!$res){echo 'process {$i} unable to lock lock. n'; }else{echo 'process {$i} successfully got the lock n';$lock->unlock(); } exit();}

這時候兩個進程都能得到鎖。

sysvsem模塊中的信號量

sem_get創建信號量

sem_remove刪除信號量(一般不用)

sem_acquire請求得到信號量

sem_release釋放信號量。和sem_acquire成對使用。

$key = ftok(’/tmp’, ’c’);$sem = sem_get($key);for($i=0; $i<2; $i++){ $pid = pcntl_fork(); if($pid <0){die('fork failed'); }elseif ($pid>0){//echo 'parent process n'; }else{echo 'child process {$i} is born. n';obtainLock($sem, $i); }}while (pcntl_waitpid(0, $status) != -1) { $status = pcntl_wexitstatus($status); echo 'Child $status completedn'; }sem_remove($sem); // finally remove the semfunction obtainLock ($sem, $i){ echo 'process {$i} is getting the sem n'; $res = sem_acquire($sem, true); sleep(1); if (!$res){echo 'process {$i} unable to get sem. n'; }else{echo 'process {$i} successfully got the sem n';sem_release($sem); } exit();}

這里有一個問題,sem_acquire()第二個參數$nowait默認為false,阻塞。我設為了true,如果得到鎖失敗,那么后面的sem_release會報警告PHP Warning: sem_release(): SysV semaphore 4 (key 0x63000081) is not currently acquired in /home/jason/sysvsem.php on line 33, 所以這里的release操作必須放在得到鎖的情況下執行,前面的幾個例子中沒有這個問題,沒得到鎖執行release也不會報錯。當然最好還是成對出現,確保得到鎖的情況下再release。

此外,ftok這個方法的參數有必要說明下,第一個 必須是existing, accessable的文件, 一般使用項目中的文件,第二個是單字符字符串。返回一個int。

輸出為

parent process parent process child process 1 is born. process 1 is getting the mutex child process 0 is born. process 0 is getting the mutex process 1 successfully got the mutex Child 0 completedprocess 0 unable to lock mutex. Child 0 completed

最后,如果文中有錯誤的地方,希望大神指出,幫助一下菜鳥進步,謝謝各位。

標簽: PHP
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日韩欧美中文在线观看| 日韩精品久久理论片| 久久国产精品美女| 国产精品一区亚洲| 麻豆一区在线| 久久天堂av| 欧美99久久| 免费久久99精品国产| 在线亚洲欧美| 亚洲精品影院在线观看| 国产亚洲久久| av中文字幕在线观看第一页| 久久国产中文字幕| 亚洲有吗中文字幕| 欧美aaaaaa午夜精品| av在线日韩| 日韩三级精品| 亚洲播播91| 日韩成人av影视| 国产黄大片在线观看| 亚洲视频播放| 久久久精品区| 亚洲中午字幕| 91综合久久爱com| 久久久精品午夜少妇| 日韩福利视频网| 国产精品99视频| 日本中文字幕不卡| 伊人精品一区| 精品中国亚洲| 亚洲ww精品| 好吊日精品视频| 欧美韩日一区| 日韩超碰人人爽人人做人人添| 亚洲日本网址| 久久精品伊人| 欧美日韩 国产精品| 午夜精品免费| 久久久久亚洲精品中文字幕| 丝袜美腿亚洲一区| 91精品蜜臀一区二区三区在线| 日韩国产91| 亚洲一区二区成人| 久久高清免费| 中文字幕系列一区| 国产一区二区三区久久| 国产美女撒尿一区二区| 中文字幕亚洲影视| 日韩午夜av| 麻豆精品网站| 黄色成人91| 午夜精品亚洲| 欧美日韩水蜜桃| 在线中文字幕播放| 亚洲精品**中文毛片| 久久97久久97精品免视看秋霞| 日韩欧美久久| 亚洲ab电影| 91av一区| 国产精品自在| 精品日本视频| 欧美国产偷国产精品三区| 精品国产aⅴ| 欧美丰满日韩| 天堂网在线观看国产精品| 久久一区二区中文字幕| 播放一区二区| 男人天堂欧美日韩| 视频一区在线视频| 91欧美极品| 四虎国产精品免费观看| 99精品电影| 亚洲一区二区三区中文字幕在线观看| 免费日韩av| 久久成人高清| 久久久久午夜电影| 免费成人在线影院| 国产精品亚洲成在人线| 91综合视频| 黄色免费成人| 欧美一区91| 日韩精品2区| 爽好久久久欧美精品| 亚洲小说春色综合另类电影| 久久精品凹凸全集| 日韩高清中文字幕一区二区| 男女性色大片免费观看一区二区| 欧美一区自拍| 欧美/亚洲一区| 欧美成人精品一级| 午夜精品亚洲| 国内不卡的一区二区三区中文字幕| 亚洲天堂1区| 国产欧美大片| 免费成人在线视频观看| 美女av在线免费看| 亚久久调教视频| 日韩在线观看不卡| 日韩精品免费一区二区夜夜嗨 | 91福利精品在线观看| 99精品电影| 国产一区不卡| 欧美亚洲免费| 亚洲尤物av| 在线视频精品| 久久精品国语| 国产成人调教视频在线观看| 亚洲精品四区| 免费看日韩精品| 欧美日韩国产综合网| 国产99在线| 国产调教精品| 青青伊人久久| 亚洲+小说+欧美+激情+另类| 国产一区导航| 国产午夜久久| 免费观看久久久4p| 亚欧成人精品| 欧美伊人久久| 国产精品极品在线观看| 国产欧美自拍| 国产精品亚洲二区| 另类小说一区二区三区| 国产精品欧美大片| 国产一区二区精品久| 国产精品99一区二区三区| 国产伦精品一区二区三区视频| 青青草国产成人99久久| 国产激情欧美| 丁香婷婷久久| 精品精品久久| japanese国产精品| 久久福利一区| 欧美亚洲福利| 国产精品99视频| 国产亚洲永久域名| 久久精品凹凸全集| av资源亚洲| 亚洲视频播放| 国产精品3区| 99久久视频| 视频一区中文| 日韩精品一页| 日韩视频网站在线观看| 美国三级日本三级久久99| 国产日韩视频在线| 日本在线精品| 国产欧美自拍| av不卡在线| 国产不卡av一区二区| 99国产精品| 超碰99在线| 久久成人亚洲| sm捆绑调教国产免费网站在线观看 | 亚洲黄页一区| 蜜桃久久久久| 丝袜美腿亚洲色图| 色偷偷色偷偷色偷偷在线视频| 亚洲精品麻豆| 欧美日韩中文一区二区| 国产精品久久久久久久免费软件| 亚洲电影在线一区二区三区| 国产精品一区二区中文字幕| 午夜久久久久| av高清不卡| 久久久91麻豆精品国产一区| 伊人久久成人| 国产欧美一区二区三区精品酒店| 日韩欧美中文字幕一区二区三区 | 亚洲精品国产偷自在线观看| 欧美国产中文高清| 日韩欧美2区| 亚洲欧美日本日韩| 国产99精品一区| 欧美日韩视频网站| 狠狠久久伊人| 国产乱码精品一区二区亚洲| 亚洲制服欧美另类| 亚洲激情国产| 成人精品亚洲| 久久影院午夜精品| 国产精品日韩精品在线播放| 日韩国产一区二| 久久狠狠亚洲综合| 欧美欧美黄在线二区| 欧美一级二级三级视频| 日韩精品欧美大片| **爰片久久毛片| 国产精品资源| 日本一二区不卡| 国产96在线亚洲| 欧洲av不卡| 亚洲夜间福利| 乱人伦精品视频在线观看| 免费在线视频一区| 日韩av网站免费在线| 日韩在线黄色| 欧美激情福利| 久久久久久色 | 国产不卡av一区二区|