摘要:請注意,此函數(shù)返回一個編號,其他函數(shù)可使用該編號操作該共享內(nèi)存段。從內(nèi)存段讀取數(shù)據(jù)從共享內(nèi)存段讀取數(shù)據(jù)很簡單。函數(shù)將該內(nèi)存段標(biāo)記為刪除,阻止任何其他進(jìn)程打開它。
在之前的一篇博客[了解一下共享內(nèi)存的概念及優(yōu)缺點(diǎn)]已經(jīng)對共享內(nèi)存的概念做了說明。下面就來簡單使用共享內(nèi)存(其實(shí)也可以用其他工具,比如redis)
PHP做內(nèi)存共享有兩套接口。一個是shm,它實(shí)際上是變量共享,會把對象變量序列化后再儲存。使用起來倒是挺方便,但是序列化存儲對于效率優(yōu)先的內(nèi)存訪問操作而言就沒啥意義了。另外一個是shmop,它是Linux和Windows通用的,不過功能上比shm弱了一些,在 Linux 上,這些函數(shù)直接是通過調(diào)用 shm* 系列的函數(shù)實(shí)現(xiàn),而 Winodows 上也通過對系統(tǒng)函數(shù)的封裝實(shí)現(xiàn)了同樣的調(diào)用。我這里先用的是shmop。
要創(chuàng)建共享內(nèi)存段需要使用函數(shù)shmop,那么前提需要開啟擴(kuò)展,可以參考[給PHP開啟shmop擴(kuò)展實(shí)現(xiàn)共享內(nèi)存].
shmop主要函數(shù)
shmop_open (創(chuàng)建或打開共享內(nèi)存塊)、shmop_write (向共享內(nèi)存塊中寫入數(shù)據(jù))、shmop_read (從共享內(nèi)存塊中讀取數(shù)據(jù))、shmop_size (獲取共享內(nèi)存塊的大?。?、shmop_close (關(guān)閉共享內(nèi)存塊)、shmop_delete (刪除共享內(nèi)存塊)
shmop_open(創(chuàng)建內(nèi)存段)
該函數(shù)中出現(xiàn)的第一個事物是系統(tǒng) ID 參數(shù)。這是標(biāo)識系統(tǒng)中的共享內(nèi)存段的數(shù)字。第二個參數(shù)是訪問模式,它非常類似于 fopen 函數(shù)的訪問模式。您可以在 4 種不同的模式下訪問一個內(nèi)存段:
模式 “a”,它允許您訪問只讀內(nèi)存段,只讀訪問
模式 “w”,它允許您訪問可讀寫的內(nèi)存段,讀寫
模式 “c”,它創(chuàng)建一個新內(nèi)存段,或者如果該內(nèi)存段已存在,嘗試打開它進(jìn)行讀寫
模式 “n”,它創(chuàng)建一個新內(nèi)存段,如果同樣 key 的已存在,則會創(chuàng)建失敗,這是為了安全使用共享內(nèi)存考慮。
第三個參數(shù)是內(nèi)存段的權(quán)限。您必須在這里提供一個八進(jìn)制值。第四個參數(shù)提供內(nèi)存段大小,以字節(jié)為單位。由于使用的共享內(nèi)存片段是固定長度的,在存儲和讀取的時候要計算好數(shù)據(jù)的長度,不然可能會寫入失敗或者讀取空值。。
請注意,此函數(shù)返回一個 ID 編號,其他函數(shù)可使用該 ID 編號操作該共享內(nèi)存段。這個 ID 是共享內(nèi)存訪問 ID,與系統(tǒng) ID 不同,它以參數(shù)的形式傳遞。請注意不要混淆這兩者。如果失敗,shmop_open 將返回 FALSE。在創(chuàng)建內(nèi)存塊時建議key參數(shù)用常量而不用變量,否則很有可能造成內(nèi)存泄露。
shmop_write(向內(nèi)存段寫入數(shù)據(jù))
這個函數(shù)類似于 fwrite 函數(shù),后者有兩個參數(shù):打開的流資源(由 fopen 返回)和您希望寫入的數(shù)據(jù)。shmop_write 函數(shù)也執(zhí)行此任務(wù)。
第一個參數(shù)是 shmop_open 返回的 ID,它識別您操作的共享內(nèi)存塊。第二個參數(shù)是您希望存儲的數(shù)據(jù),最后的第三個參數(shù)是您希望開始寫入的位置。默認(rèn)情況下,我們始終使用 0 來表示開始寫入的位置。請注意,此函數(shù)在失敗時會返回 FALSE,在成功時會返回寫入的字節(jié)數(shù)。
shmop_read(從內(nèi)存段讀取數(shù)據(jù))
從共享內(nèi)存段讀取數(shù)據(jù)很簡單。您只需要一個打開的內(nèi)存段和 shmop_read 函數(shù)。此函數(shù)接受一些參數(shù),工作原理類似于 fread。
請留意這里的參數(shù)。shmop_read 函數(shù)將接受 shmop_open 返回的 ID,我們已知道它,不過它還接受另外兩個參數(shù)。第二個參數(shù)是您希望從內(nèi)存段讀取的位置,而第三個是您希望讀取的字節(jié)數(shù)。第二個參數(shù)可以始終為 0,表示數(shù)據(jù)的開頭,但第三個參數(shù)可能存在問題,因?yàn)槲覀儾恢牢覀兿Mx取多少字節(jié)。
這非常類似于我們在 fread 函數(shù)中的行為,該函數(shù)接受兩個參數(shù):打開的流資源(由 fopen 返回)和您希望從該流讀取的字節(jié)數(shù)。使用 filesize 函數(shù)(它返回一個文件中的字節(jié)數(shù))來完整地讀取它。
shmop_size(返回內(nèi)存段數(shù)據(jù)實(shí)際大?。?/em>
比如,我們開辟了一個長度為100字節(jié)的內(nèi)存空間,但是實(shí)際存入的數(shù)據(jù)長度僅僅90,那么使用shmop_size返回的值就是90.
shmop_delete(刪除內(nèi)存段)
該函數(shù)僅接受一個參數(shù):我們希望刪除的共享內(nèi)存 ID,這不會實(shí)際刪除該內(nèi)存段。它將該內(nèi)存段標(biāo)記為刪除,因?yàn)楣蚕韮?nèi)存段在有其他進(jìn)程正在使用它時無法被刪除。shmop_delete 函數(shù)將該內(nèi)存段標(biāo)記為刪除,阻止任何其他進(jìn)程打開它。要刪除它,我們需要關(guān)閉該內(nèi)存段。在創(chuàng)建內(nèi)存塊時建議key參數(shù)用常量而不用變量,否則很有可能造成內(nèi)存泄露。
shmop_close(關(guān)閉內(nèi)存段)
我們在對內(nèi)存段進(jìn)行讀取和寫入,但完成操作后,我們必須從它解除,這非常類似于處理文件時的 fclose 函數(shù)。打開包含一個文件的流并在其中讀取或?qū)懭霐?shù)據(jù)后,我們必須關(guān)閉它,否則將發(fā)生鎖定。
簡單測試結(jié)果查看
我是在LNMP環(huán)境下操作的,如果你也和我一樣,在執(zhí)行完簡單的操作之后,可以使用linux命令查看一下地址和占用大小
# ipcs -m[root@bogon ~]# ipcs -m ------ Shared Memory Segments -------- key shmid owner perms bytes nattch status 0x00000000 0 gdm 600 393216 2 dest 0x00000000 32769 gdm 600 393216 2 dest 0x4337b101 884750 nobody 644 1024 0命令說明
key :共享內(nèi)存的唯一的key值,共享內(nèi)存通過該key來判斷你讀取的是哪一塊內(nèi)存。
shmid:當(dāng)使用key來獲取內(nèi)存時,你獲得的是這個id的值。它作為你操作內(nèi)存塊的標(biāo)識。
owner:創(chuàng)建該共享內(nèi)存塊的用戶
perms:該共享內(nèi)存的讀寫權(quán)限,8禁止,可以是777,與文件的讀寫權(quán)限一致。
bytes:該內(nèi)存塊的大小
nattch:連接該內(nèi)存塊的進(jìn)程數(shù)
status:當(dāng)前狀態(tài),如:dest,即將刪除等。項目實(shí)際應(yīng)用小案例
_memory_key, "a", 0644, $this->_memory_size); if ($shmid === FALSE) { $shmid = @shmop_open($this->_memory_key, "c", 0644, $this->_memory_size); $data = $this->return_skill_list(); shmop_write($shmid, json_encode($data), 0); @shmop_close($shmid); return $data; } $data = json_decode(preg_replace("/[x00-x1Fx80-x9F]/u", "", trim(shmop_read($shmid, 0, $this->_memory_size))), true); @shmop_close($shmid); return $data; } public function return_skill_list() { //這里是一個超大的數(shù)組,其實(shí)就是把這個數(shù)組json化,然后存入共享內(nèi)存段。 其實(shí)可以用redis等其他緩存...這里我就是為了不用redis等其他nosql才用的shmop return array ( 1 => array ("id" => "1","animation" => "13","skill_type" => "1","power_type" => "1","site" => "1","type" => "1","paramete" => "0","paramete2" => "0","paramete3" => "0","chance" => "0","ratio" => "1", ), 2 => array ("id" => "2","animation" => "3","skill_type" => "2","power_type" => "1","site" => "1","type" => "1","paramete" => "0","paramete2" => "0","paramete3" => "0","chance" => "0","ratio" => "2", ),..........................................當(dāng)然你要考慮的是,如果數(shù)據(jù)更新的話,那么內(nèi)存段也要刪除,并且更新數(shù)據(jù)......................通過shmop_delete可以刪除 。這就需要你們自己根據(jù)項目應(yīng)用來考慮了
還有就是我這里只是為了簡單的讀,并沒有出現(xiàn)復(fù)雜的讀寫,否則可能會出現(xiàn)進(jìn)程互斥等意想不到的沖突~如果復(fù)雜,那么就可以考慮信號量了~
如果你在項目中遇到主從場景??梢詤⒖歼@篇文章 http://tubaluer.iteye.com/blo...
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/28939.html
摘要:請注意,此函數(shù)返回一個編號,其他函數(shù)可使用該編號操作該共享內(nèi)存段。從內(nèi)存段讀取數(shù)據(jù)從共享內(nèi)存段讀取數(shù)據(jù)很簡單。函數(shù)將該內(nèi)存段標(biāo)記為刪除,阻止任何其他進(jìn)程打開它。 在之前的一篇博客[了解一下共享內(nèi)存的概念及優(yōu)缺點(diǎn)]已經(jīng)對共享內(nèi)存的概念做了說明。下面就來簡單使用共享內(nèi)存(其實(shí)也可以用其他工具,比如redis) PHP做內(nèi)存共享有兩套接口。一個是shm,它實(shí)際上是變量共享,會把對象變量序列化...
摘要:如果是這樣,就需要引入信號量進(jìn)行控制。這應(yīng)該是因?yàn)檎诠蚕韮?nèi)存被上一次操作占用中還沒有釋放導(dǎo)致。 共享內(nèi)存 共享內(nèi)存的使用主要是為了能夠在同一臺機(jī)器不同的進(jìn)程中共享一些數(shù)據(jù),比如在多個 php-fpm 進(jìn)程中共享當(dāng)前進(jìn)程的使用情況。這種通信也稱為進(jìn)程間通信(Inter-Process Communication),簡稱 IPC。 PHP 內(nèi)置的 shmop 擴(kuò)展 (Shared Mem...
摘要:是一個較小的抽象層,用于使用操作共享內(nèi)存,支持以一種面向?qū)ο蟮姆绞捷p松操作內(nèi)存段。在編寫使用共享內(nèi)存進(jìn)行存儲的小型應(yīng)用程序時,這個庫可幫助創(chuàng)建非常簡潔的代碼。不要低估共享內(nèi)存在應(yīng)用程序中的力量。 SimpleSHM 是一個較小的抽象層,用于使用 PHP 操作共享內(nèi)存,支持以一種面向?qū)ο蟮姆绞捷p松操作內(nèi)存段。在編寫使用共享內(nèi)存進(jìn)行存儲的小型應(yīng)用程序時,這個庫可幫助創(chuàng)建非常簡潔的代碼。可以...
閱讀 3020·2021-10-12 10:12
閱讀 3073·2021-09-22 16:04
閱讀 3306·2019-08-30 15:54
閱讀 2616·2019-08-29 16:59
閱讀 2929·2019-08-29 16:08
閱讀 880·2019-08-29 11:20
閱讀 3503·2019-08-28 18:08
閱讀 661·2019-08-26 13:43