国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁技術(shù)文章
文章詳情頁

PHP編程中的鎖

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

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

文件鎖

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

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

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

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

發(fā)送USR2信號給pid文件記錄的進(jìn)程,信號屬于進(jìn)程通信,會另開一個篇幅。

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

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

$operation是操作類型。

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

需要注意的是,這個函數(shù)默認(rèn)是阻塞的,如果想非阻塞可以在 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

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

sync模塊中的讀寫鎖

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

sync模塊中的Event

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

上測試代碼

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() 一次只喚醒一個進(jìn)程。

pthreads模塊

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

信號量sync模塊中的信號量

SyncSemaphore文檔中顯示,它和Mutex的不同之處,在于Semaphore一次可以被多個進(jìn)程(或線程)得到,而Mutex一次只能被一個得到。所以在SyncSemaphore的構(gòu)造函數(shù)中,有一個參數(shù)指定信號量可以被多少進(jìn)程得到。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();}

這時候兩個進(jìn)程都能得到鎖。

sysvsem模塊中的信號量

sem_get創(chuàng)建信號量

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()第二個參數(shù)$nowait默認(rèn)為false,阻塞。我設(shè)為了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操作必須放在得到鎖的情況下執(zhí)行,前面的幾個例子中沒有這個問題,沒得到鎖執(zhí)行release也不會報錯。當(dāng)然最好還是成對出現(xiàn),確保得到鎖的情況下再release。

此外,ftok這個方法的參數(shù)有必要說明下,第一個 必須是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

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

標(biāo)簽: PHP
相關(guān)文章:
主站蜘蛛池模板: 91寡妇天天综合久久影院 | 国产成人刺激视频在线观看 | 免费黄色三级网站 | av狼论坛| 成人老司机深夜福利久久 | 国产视频高清在线 | 中文字幕在线看片成人 | 有码 在线 | 欧美日韩精品一区二区三区 | 国产萌白酱在线一区二区 | 日本三级韩国三级在线观看a级 | 国产乱子伦露脸对白在线小说 | 久久99国产亚洲精品观看 | 欧美一级大黄特黄毛片视频 | 国产日产久久高清欧美一区 | 怡红院免费全部视频在线视频 | 亚洲欧美另类色妞网站 | 三级网站在线免费观看 | 全部毛片| 国产东北色老头老太性视频 | 男人精品一线视频在线观看 | 国产乱码一区二区三区四川人 | 国产亚洲欧美一区二区 | 国产一区精品在线 | 91久久香蕉国产线看观看软件 | 国产性做久久久久久 | 亚洲日韩中文字幕天堂不卡 | 国内精品小视频在线 | 在线观看日本亚洲一区 | 国产欧美一区二区精品久久久 | 国产精品免费看久久久久 | 日韩精品一区二区三区 在线观看 | 亚洲国产欧美91 | 欧美精品videos | 猛操美女 | 99热只有精品一区二区 | 免费观看欧美成人h | 亚洲在线网| 最新福利片v国产片 | 中国成人在线视频 | 亚洲成年男人的天堂网 |