前言
進(jìn)程間通信是一個(gè)永遠(yuǎn)的話題,我的上一篇文章通過(guò)一個(gè)并發(fā)循環(huán)ID生成器的實(shí)現(xiàn)介紹了如何使用外部介質(zhì)來(lái)進(jìn)行進(jìn)程間通信:從并發(fā)處理談PHP進(jìn)程間通信(一)外部介質(zhì) 。介紹的幾種方法適用于各種語(yǔ)言,但是他們都依賴于一種外部介質(zhì),文化的讀寫(xiě)有瓶頸,mysql 和 redis 會(huì)掛掉或連接超時(shí),歸根結(jié)底總覺(jué)得在 HACK;
對(duì)于進(jìn)程間通信,每一個(gè)完備的語(yǔ)言都應(yīng)該有對(duì)應(yīng)的處理方式,而 PHP 對(duì)應(yīng)的則是一族對(duì) UNIX SYSTEM V包裝的函數(shù),包括信號(hào)量(semaphore)、共享內(nèi)存(shared memory)和消息隊(duì)列(msg queue)的操作。
它的安裝和使用非常簡(jiǎn)單,在編譯 PHP 時(shí)添加 --enable-sysvsem --enable-sysvshm --enable-sysvmsg
參數(shù)就可以,當(dāng)然 Windows 上無(wú)法使用。
今天我們?nèi)耘f使用上一篇文章的例子來(lái)介紹 PHP 內(nèi)部實(shí)現(xiàn)的進(jìn)程間通信,在了解它們的具體使用之前,先簡(jiǎn)單介紹一下信號(hào)量、共享內(nèi)存、消息隊(duì)列的概念。
Unix System V IPC
信號(hào)量
信號(hào)量又稱(chēng)為信號(hào)燈,它是用來(lái)協(xié)調(diào)不同進(jìn)程間的數(shù)據(jù)對(duì)象的,而最主要的應(yīng)用是共享內(nèi)存方式的進(jìn)程間通信。本質(zhì)上,信號(hào)量是一個(gè)計(jì)數(shù)器,它用來(lái)記錄對(duì)某個(gè)資源(如共享內(nèi)存)的存取狀況。
一般說(shuō)來(lái),為了獲得共享資源,進(jìn)程需要執(zhí)行下列操作:
獲取控制共享資源的信號(hào)量的值;
若值為正,進(jìn)程將信號(hào)量減1,進(jìn)程操作共享資源,進(jìn)入步驟4;
若值0,則拒絕進(jìn)程使用共享資源,進(jìn)程進(jìn)入睡眠狀態(tài),直至信號(hào)量值大于0后,進(jìn)程被喚醒,轉(zhuǎn)入步驟1;
當(dāng)進(jìn)程不再使用共享資源時(shí),將信號(hào)量值加1。如果此時(shí)有進(jìn)程正在睡眠等待此信號(hào)量,則喚醒此進(jìn)程;
信號(hào)量的使用可以類(lèi)比為:
一個(gè)房間必須用鑰匙才能開(kāi)門(mén),有N把鑰匙放在門(mén)口,拿到鑰匙開(kāi)門(mén)進(jìn)入房間,出來(lái)時(shí)將鑰匙放回并告知等待的人去取鑰匙開(kāi)門(mén)。 此例中,鑰匙的數(shù)量限制了同一時(shí)間內(nèi)在房間的最大人數(shù)。房間即共享資源,鑰匙是信號(hào)量,而想進(jìn)入房間的人則是多個(gè)進(jìn)程。
信號(hào)量有二值和多值之分,一般共享資源都不允許多