午夜剧场伦理_日本一道高清_国产又黄又硬_91黄色网战_女同久久另类69精品国产_妹妹的朋友在线

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

PHP編程中的鎖

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

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

文件鎖

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

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

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

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

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

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

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

$operation是操作類(lèi)型。

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

需要注意的是,這個(gè)函數(shù)默認(rèn)是阻塞的,如果想非阻塞可以在 operation 加一個(gè) bitmaskLOCK_NB. 接下來(lái)測(cè)試一下。

$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,運(yùn)行php process.php &, 此時(shí)再次運(yùn)行php process.php,就可以看到錯(cuò)誤提示。flock也有共享鎖,LOCK_SH.

互斥鎖和讀寫(xiě)鎖sync模塊中的Mutex

Mutex是一個(gè)組合詞,mutual exclusion。用pecl安裝一下sync模塊,pecl install sync。 文檔中的SyncMutex只有兩個(gè)方法,lock 和 unlock, 我們就直接上代碼測(cè)試吧。沒(méi)有用IDE寫(xiě),所以cs異常丑陋,請(qǐng)無(wú)視。

$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不一定誰(shuí)在前面。但是總有一個(gè)得不到鎖。這里SyncMutex::lock(int $millisecond)的參數(shù)是 millisecond, 代表阻塞的時(shí)長(zhǎng), -1 為無(wú)限阻塞。

sync模塊中的讀寫(xiě)鎖

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

sync模塊中的Event

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

上測(cè)試代碼

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();}

這里故意少寫(xiě)一個(gè)fire(), 所以程序會(huì)阻塞,證明了 fire() 一次只喚醒一個(gè)進(jìn)程。

pthreads模塊

貌似也看到了Mutex, Cond, Pool. 沒(méi)來(lái)得及看,看完再補(bǔ)充。

信號(hào)量sync模塊中的信號(hào)量

SyncSemaphore文檔中顯示,它和Mutex的不同之處,在于Semaphore一次可以被多個(gè)進(jìn)程(或線程)得到,而Mutex一次只能被一個(gè)得到。所以在SyncSemaphore的構(gòu)造函數(shù)中,有一個(gè)參數(shù)指定信號(hào)量可以被多少進(jìn)程得到。public SyncSemaphore::__construct ([ string $name [, integer $initialval [, bool $autounlock ]]] )就是這個(gè)$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();}

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

sysvsem模塊中的信號(hào)量

sem_get創(chuàng)建信號(hào)量

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

sem_acquire請(qǐng)求得到信號(hào)量

sem_release釋放信號(hào)量。和sem_acquire成對(duì)使用。

$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();}

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

此外,ftok這個(gè)方法的參數(shù)有必要說(shuō)明下,第一個(gè) 必須是existing, accessable的文件, 一般使用項(xiàng)目中的文件,第二個(gè)是單字符字符串。返回一個(gè)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

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

標(biāo)簽: PHP
相關(guān)文章:
主站蜘蛛池模板: 欧美日韩国产在线一区 | 欧美特级特黄aaaaaa在线看 | 欧洲在线 | 日韩精品社区 | 99成人精品 | 日本中文字幕精品 | 在线欧美成人 | 国产91精品一区二区 | www超碰在线 | 国产精品婷婷 | 中文在线字幕观看 | 亚洲最大av网 | 日韩欧美网 | 黄色免费在线观看网站 | 日韩国产在线播放 | 午夜精品剧场 | 久久久午夜影院 | 69xxx免费 | 在线黄色免费 | 二区三区在线观看 | 国产欧美高清 | 欧美日韩在线观看一区二区 | 欧美一级免费观看 | 成人av午夜 | 免费看的毛片 | 激情五月综合网 | www婷婷| 污网站在线免费看 | 国产一级片免费看 | 一级黄色性生活片 | 日韩成人精品一区二区 | 日韩 欧美 综合 | 成人国产视频在线观看 | 91视频高清 | 久久影院中文字幕 | 91精品免费视频 | 免费看一级黄色片 | 欧美一级片网站 | 美女视频一区 | 国产一区二区视频免费观看 | 欧美中文日韩 |