php簡單使用shmop函數建立共用記憶體減少伺服器負載

來源:互聯網
上載者:User
這篇文章主要介紹了關於php簡單使用shmop函數建立共用記憶體減少伺服器負載,有著一定的參考價值,現在分享給大家,有需要的朋友可以參考一下

在之前的一篇部落格[瞭解一下共用記憶體的概念及優缺點]已經對共用記憶體的概念做了說明。下面就來簡單使用共用記憶體(其實也可以用其他工具,比如redis)

PHP做記憶體共用有兩套介面。一個是shm,它實際上是變數共用,會把物件變數序列化後再儲存。使用起來倒是挺方便,但是序列化儲存對於效率優先的記憶體訪問操作而言就沒啥意義了。另外一個是shmop,它是Linux和Windows通用的,不過功能上比shm弱了一些,在 Linux 上,這些函數直接是通過調用 shm* 系列的函數實現,而 Winodows 上也通過對系統函數的封裝實現了同樣的調用。我這裡先用的是shmop。

要建立共用記憶體段需要使用函數shmop,那麼前提需要開啟擴充,可以參考[給PHP開啟shmop擴充實現共用記憶體].

shmop主要函數

shmop_open (建立或開啟共用記憶體塊)、shmop_write (向共用記憶體塊中寫入資料)、shmop_read (從共用記憶體塊中讀取資料)、shmop_size (擷取共用記憶體塊的大小)、shmop_close (關閉共用記憶體塊)、shmop_delete (刪除共用記憶體塊)

<?php//建立一塊共用記憶體$shm_key = 0x4337b101;$shm_id = @shmop_open($shm_key, 'c', 0644, 1024);//讀取並寫入資料$data = shmop_read($shm_id, 0, 1024);shmop_write($shm_id, json_encode($data), 0);$size = shmop_size($shm_id);  //擷取記憶體中實際資料佔用大小//關閉記憶體塊,並不會刪除共用記憶體,只是清除 PHP 的資源shmop_close($shm_id);

shmop_open(建立記憶體段)

該函數中出現的第一個事物是系統 ID 參數。這是標識系統中的共用記憶體段的數字。第二個參數是訪問模式,它非常類似於 fopen 函數的訪問模式。您可以在 4 種不同的模式下訪問一個記憶體段:

模式 “a”,它允許您訪問唯讀記憶體段,唯讀訪問
模式 “w”,它允許您訪問可讀寫的記憶體段,讀寫
模式 “c”,它建立一個新記憶體段,或者如果該記憶體段已存在,嘗試開啟它進行讀寫
模式 “n”,它建立一個新記憶體段,如果同樣 key 的已存在,則會建立失敗,這是為了安全使用共用記憶體考慮。
第三個參數是記憶體段的許可權。您必須在這裡提供一個八進位值。

第四個參數提供記憶體段大小,以位元組為單位。由於使用的共用記憶體片段是固定長度的,在儲存和讀取的時候要計算好資料的長度,不然可能會寫入失敗或者讀取空值。。

請注意,此函數返回一個 ID 編號,其他函數可使用該 ID 編號操作該共用記憶體段。這個 ID 是共用記憶體訪問 ID,與系統 ID 不同,它以參數的形式傳遞。請注意不要混淆這兩者。如果失敗,shmop_open 將返回 FALSE。在建立記憶體塊時建議key參數用常量而不用變數,否則很有可能造成記憶體泄露。

shmop_write(向記憶體段寫入資料)

這個函數類似於 fwrite 函數,後者有兩個參數:開啟的流資源(由 fopen 返回)和您希望寫入的資料。shmop_write 函數也執行此任務。

第一個參數是 shmop_open 返回的 ID,它識別您操作的共用記憶體塊。第二個參數是您希望儲存的資料,最後的第三個參數是您希望開始寫入的位置。預設情況下,我們始終使用 0 來表示開始寫入的位置。請注意,此函數在失敗時會返回 FALSE,在成功時會返回寫入的位元組數。

shmop_read(從記憶體段讀取資料)

從共用記憶體段讀取資料很簡單。您只需要一個開啟的記憶體段和 shmop_read 函數。此函數接受一些參數,工作原理類似於 fread。

請留意這裡的參數。shmop_read 函數將接受 shmop_open 返回的 ID,我們已知道它,不過它還接受另外兩個參數。第二個參數是您希望從記憶體段讀取的位置,而第三個是您希望讀取的位元組數。第二個參數可以始終為 0,表示資料的開頭,但第三個參數可能存在問題,因為我們不知道我們希望讀取多少位元組。

這非常類似於我們在 fread 函數中的行為,該函數接受兩個參數:開啟的流資源(由 fopen 返回)和您希望從該流讀取的位元組數。使用 filesize 函數(它返回一個檔案中的位元組數)來完整地讀取它。

shmop_size(返回記憶體段資料實際大小)

比如,我們開闢了一個長度為100位元組的記憶體空間,但是實際存入的資料長度僅僅90,那麼使用shmop_size返回的值就是90.

shmop_delete(刪除記憶體段)

該函數僅接受一個參數:我們希望刪除的共用記憶體 ID,這不會實際刪除該記憶體段。它將該記憶體區段標記為刪除,因為共用記憶體段在有其他進程正在使用它時無法被刪除。shmop_delete 函數將該記憶體區段標記為刪除,阻止任何其他進程開啟它。要刪除它,我們需要關閉該記憶體段。在建立記憶體塊時建議key參數用常量而不用變數,否則很有可能造成記憶體泄露。

shmop_close(關閉記憶體段)

我們在對記憶體段進行讀取和寫入,但完成操作後,我們必須從它解除,這非常類似於處理檔案時的 fclose 函數。開啟包含一個檔案的流並在其中讀取或寫入資料後,我們必須關閉它,否則將發生鎖定。

簡單測試結果查看

我是在LNMP環境下操作的,如果你也和我一樣,在執行完簡單的操作之後,可以使用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 :共用記憶體的唯一的key值,共用記憶體通過該key來判斷你讀取的是哪一塊記憶體。
shmid:當使用key來擷取記憶體時,你獲得的是這個id的值。它作為你操作記憶體塊的標識。
owner:建立該共用記憶體塊的使用者
perms:該共用記憶體的讀寫權限,8禁止,可以是777,與檔案的讀寫權限一致。
bytes:該記憶體塊的大小
nattch:串連該記憶體塊的進程數
status:目前狀態,如:dest,即將刪除等。

項目實際應用小案例

<?php/** * 將領技能 */class Generalskill_model extends CI_Model {    private $_memory_key = 0x4337b001;   //共用記憶體位址key  private $_memory_size = 1048576;     //開闢共用記憶體大小  //最好根據實際資料長度大小定義。  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() {   //這裡是一個超大的數組,其實就是把這個數組json化,然後存入共用記憶體段。  其實可以用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',  ),..........................................

當然你要考慮的是,如果資料更新的話,那麼記憶體段也要刪除,並且更新資料......................通過shmop_delete可以刪除 。這就需要你們自己根據項目應用來考慮了

還有就是我這裡只是為了簡單的讀,並沒有出現複雜的讀寫,否則可能會出現進程互斥等意想不到的衝突~如果複雜,那麼就可以考慮訊號量了~

以上就是本文的全部內容,希望對大家的學習有所協助,更多相關內容請關注topic.alibabacloud.com!

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.