主頁(yè) > 知識(shí)庫(kù) > php使用shmop函數(shù)創(chuàng)建共享內(nèi)存減少負(fù)載的方法

php使用shmop函數(shù)創(chuàng)建共享內(nèi)存減少負(fù)載的方法

熱門標(biāo)簽:福州人工外呼系統(tǒng)哪家強(qiáng) 安裝電銷外呼系統(tǒng) 百度商鋪地圖標(biāo)注 注冊(cè)400電話申請(qǐng) 衡水外呼系統(tǒng)平臺(tái) 釘釘打卡地圖標(biāo)注 常州地圖標(biāo)注服務(wù)商 新河科技智能外呼系統(tǒng)怎么樣 地圖標(biāo)注平臺(tái)怎么給錢注冊(cè)

PHP做內(nèi)存共享有兩套接口。一個(gè)是shm,它實(shí)際上是變量共享,會(huì)把對(duì)象變量序列化后再儲(chǔ)存。使用起來(lái)倒是挺方便,但是序列化存儲(chǔ)對(duì)于效率優(yōu)先的內(nèi)存訪問(wèn)操作而言就沒(méi)啥意義了。另外一個(gè)是shmop,它是Linux和Windows通用的,不過(guò)功能上比shm弱了一些,在 Linux 上,這些函數(shù)直接是通過(guò)調(diào)用 shm* 系列的函數(shù)實(shí)現(xiàn),而 Winodows 上也通過(guò)對(duì)系統(tǒng)函數(shù)的封裝實(shí)現(xiàn)了同樣的調(diào)用。

要?jiǎng)?chuàng)建共享內(nèi)存段需要使用函數(shù)shmop,那么前提需要開啟擴(kuò)展。

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)存塊)

?php
//創(chuàng)建一塊共享內(nèi)存
$shm_key = 0x4337b101;
$shm_id = @shmop_open($shm_key, 'c', 0644, 1024);
//讀取并寫入數(shù)據(jù)
$data = shmop_read($shm_id, 0, 1024);
shmop_write($shm_id, json_encode($data), 0);
$size = shmop_size($shm_id); //獲取內(nèi)存中實(shí)際數(shù)據(jù)占用大小
//關(guān)閉內(nèi)存塊,并不會(huì)刪除共享內(nèi)存,只是清除 PHP 的資源
shmop_close($shm_id);

shmop_open(創(chuàng)建內(nèi)存段)

該函數(shù)中出現(xiàn)的第一個(gè)事物是系統(tǒng) ID 參數(shù)。這是標(biāo)識(shí)系統(tǒng)中的共享內(nèi)存段的數(shù)字。第二個(gè)參數(shù)是訪問(wèn)模式,它非常類似于 fopen 函數(shù)的訪問(wèn)模式。您可以在 4 種不同的模式下訪問(wèn)一個(gè)內(nèi)存段:

模式 “a”,它允許您訪問(wèn)只讀內(nèi)存段,只讀訪問(wèn) 模式 “c”,它創(chuàng)建一個(gè)新內(nèi)存段,或者如果該內(nèi)存段已存在,嘗試打開它進(jìn)行讀寫
模式 “n”,它創(chuàng)建一個(gè)新內(nèi)存段,如果同樣 key 的已存在,則會(huì)創(chuàng)建失敗,這是為了安全使用共享內(nèi)存考慮。

第三個(gè)參數(shù)是內(nèi)存段的權(quán)限。您必須在這里提供一個(gè)八進(jìn)制值。

第四個(gè)參數(shù)提供內(nèi)存段大小,以字節(jié)為單位。由于使用的共享內(nèi)存片段是固定長(zhǎng)度的,在存儲(chǔ)和讀取的時(shí)候要計(jì)算好數(shù)據(jù)的長(zhǎng)度,不然可能會(huì)寫入失敗或者讀取空值。。

請(qǐng)注意,此函數(shù)返回一個(gè) ID 編號(hào),其他函數(shù)可使用該 ID 編號(hào)操作該共享內(nèi)存段。這個(gè) ID 是共享內(nèi)存訪問(wèn) ID,與系統(tǒng) ID 不同,它以參數(shù)的形式傳遞。請(qǐng)注意不要混淆這兩者。如果失敗,shmop_open 將返回 FALSE。在創(chuàng)建內(nèi)存塊時(shí)建議key參數(shù)用常量而不用變量,否則很有可能造成內(nèi)存泄露。

shmop_write(向內(nèi)存段寫入數(shù)據(jù))

這個(gè)函數(shù)類似于 fwrite 函數(shù),后者有兩個(gè)參數(shù):打開的流資源(由 fopen 返回)和您希望寫入的數(shù)據(jù)。shmop_write 函數(shù)也執(zhí)行此任務(wù)。

第一個(gè)參數(shù)是 shmop_open 返回的 ID,它識(shí)別您操作的共享內(nèi)存塊。第二個(gè)參數(shù)是您希望存儲(chǔ)的數(shù)據(jù),最后的第三個(gè)參數(shù)是您希望開始寫入的位置。默認(rèn)情況下,我們始終使用 0 來(lái)表示開始寫入的位置。請(qǐng)注意,此函數(shù)在失敗時(shí)會(huì)返回 FALSE,在成功時(shí)會(huì)返回寫入的字節(jié)數(shù)。

shmop_read(從內(nèi)存段讀取數(shù)據(jù))

從共享內(nèi)存段讀取數(shù)據(jù)很簡(jiǎn)單。您只需要一個(gè)打開的內(nèi)存段和 shmop_read 函數(shù)。此函數(shù)接受一些參數(shù),工作原理類似于 fread。

請(qǐng)留意這里的參數(shù)。shmop_read 函數(shù)將接受 shmop_open 返回的 ID,我們已知道它,不過(guò)它還接受另外兩個(gè)參數(shù)。第二個(gè)參數(shù)是您希望從內(nèi)存段讀取的位置,而第三個(gè)是您希望讀取的字節(jié)數(shù)。第二個(gè)參數(shù)可以始終為 0,表示數(shù)據(jù)的開頭,但第三個(gè)參數(shù)可能存在問(wèn)題,因?yàn)槲覀儾恢牢覀兿Mx取多少字節(jié)。

這非常類似于我們?cè)?fread 函數(shù)中的行為,該函數(shù)接受兩個(gè)參數(shù):打開的流資源(由 fopen 返回)和您希望從該流讀取的字節(jié)數(shù)。使用 filesize 函數(shù)(它返回一個(gè)文件中的字節(jié)數(shù))來(lái)完整地讀取它。

shmop_size(返回內(nèi)存段數(shù)據(jù)實(shí)際大小)

比如,我們開辟了一個(gè)長(zhǎng)度為100字節(jié)的內(nèi)存空間,但是實(shí)際存入的數(shù)據(jù)長(zhǎng)度僅僅90,那么使用shmop_size返回的值就是90.

shmop_delete(刪除內(nèi)存段)

該函數(shù)僅接受一個(gè)參數(shù):我們希望刪除的共享內(nèi)存 ID,這不會(huì)實(shí)際刪除該內(nèi)存段。它將該內(nèi)存段標(biāo)記為刪除,因?yàn)楣蚕韮?nèi)存段在有其他進(jìn)程正在使用它時(shí)無(wú)法被刪除。shmop_delete 函數(shù)將該內(nèi)存段標(biāo)記為刪除,阻止任何其他進(jìn)程打開它。要?jiǎng)h除它,我們需要關(guān)閉該內(nèi)存段。在創(chuàng)建內(nèi)存塊時(shí)建議key參數(shù)用常量而不用變量,否則很有可能造成內(nèi)存泄露。

shmop_close(關(guān)閉內(nèi)存段)

我們?cè)趯?duì)內(nèi)存段進(jìn)行讀取和寫入,但完成操作后,我們必須從它解除,這非常類似于處理文件時(shí)的 fclose 函數(shù)。打開包含一個(gè)文件的流并在其中讀取或?qū)懭霐?shù)據(jù)后,我們必須關(guān)閉它,否則將發(fā)生鎖定。

簡(jiǎn)單測(cè)試結(jié)果查看

我是在LNMP環(huán)境下操作的,如果你也和我一樣,在執(zhí)行完簡(jiǎn)單的操作之后,可以使用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

命令說(shuō)明

key :共享內(nèi)存的唯一的key值,共享內(nèi)存通過(guò)該key來(lái)判斷你讀取的是哪一塊內(nèi)存。 owner:創(chuàng)建該共享內(nèi)存塊的用戶 bytes:該內(nèi)存塊的大小 status:當(dāng)前狀態(tài),如:dest,即將刪除等。

項(xiàng)目實(shí)際應(yīng)用小案例

/**
 * 將領(lǐng)技能
 */
class Generalskill_model extends CI_Model {!-- -->

 private $_memory_key = 0x4337b001; //共享內(nèi)存地址key
 private $_memory_size = 1048576;  //開辟共享內(nèi)存大小 //最好根據(jù)實(shí)際數(shù)據(jù)長(zhǎng)度大小定義。

 public function __construct() {!-- -->
 parent::__construct();
 }

 public function get_skill_list() {!-- -->
 $data = [];
 $shmid = @shmop_open($this->_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-\x1F\x80-\x9F]/u', '', trim(shmop_read($shmid, 0, $this->_memory_size))), true);
 @shmop_close($shmid);
 return $data; 

 }

 public function return_skill_list() {!-- --> //這里是一個(gè)超大的數(shù)組,其實(shí)就是把這個(gè)數(shù)組json化,然后存入共享內(nèi)存段。 其實(shí)可以用redis等其他緩存...這里我就是為了不用redis等其他nosql才用的shmop
 return array (
=> 
 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',
 ),
=> 
 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)存段也要?jiǎng)h除,并且更新數(shù)據(jù)…通過(guò)shmop_delete可以刪除 。這就需要你們自己根據(jù)項(xiàng)目應(yīng)用來(lái)考慮了

還有就是這篇文章只是為了簡(jiǎn)單的讀,并沒(méi)有出現(xiàn)復(fù)雜的讀寫,否則可能會(huì)出現(xiàn)進(jìn)程互斥等意想不到的沖突如果復(fù)雜,那么就可以考慮信號(hào)量了

到此這篇關(guān)于php使用shmop函數(shù)創(chuàng)建共享內(nèi)存減少負(fù)載的文章就介紹到這了,更多相關(guān)php創(chuàng)建共享內(nèi)存減少負(fù)載內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • PHP共享內(nèi)存使用與信號(hào)控制實(shí)例分析
  • PHP進(jìn)程通信基礎(chǔ)之信號(hào)量與共享內(nèi)存通信
  • PHP共享內(nèi)存用法實(shí)例分析
  • 單臺(tái)服務(wù)器的PHP進(jìn)程之間實(shí)現(xiàn)共享內(nèi)存的方法
  • php共享內(nèi)存段示例分享

標(biāo)簽:唐山 鷹潭 鶴崗 六安 柳州 克拉瑪依 遼陽(yáng) 白城

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《php使用shmop函數(shù)創(chuàng)建共享內(nèi)存減少負(fù)載的方法》,本文關(guān)鍵詞  php,使用,shmop,函數(shù),創(chuàng)建,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《php使用shmop函數(shù)創(chuàng)建共享內(nèi)存減少負(fù)載的方法》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于php使用shmop函數(shù)創(chuàng)建共享內(nèi)存減少負(fù)載的方法的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章