PHP支持組操作的Memcache類

Memcache是PHP開發中較常用到的緩存方法,在高並發的系統中是必不可少的組成部分。
在實際開發中,Memcache存在一個比較不盡人意的問題,就是Memcache不能支持對key進行的組操作。
 
組操作,也可以稱為域操作,比如說某個文章系統,在前臺部分使用Memcache緩存瞭列表頁數據、文章詳細頁數據,兩種數據的量都比較多。那麼,當後臺發佈瞭一篇文章的時候,列表頁就應該需要更新到最新的列表——可能涉及到許多個列表頁。當然,對文章詳細頁來說,它是不需要更新的。
 
好的,這個時候我們就需要刪除原有緩存,讓程式可以自動更新列表頁數據。但是使用Memcache的flush函數有個問題,就是它會清空全部的數據,包括列表頁和文章頁的數據,在大並發的條件下,全部緩存刪除後重建緩存的時候,將會有非常高的負載產生。
 
另外,還會有情況就是有些你不願意刪除的緩存變量,也會丟失瞭,比如說程式的配置,數據庫為瞭提速而存到緩存的表結構等。
 
所以我們需要一個支持組操作的緩存機制,我們就可以把列表頁設置成一個組,文章頁數據是另外一個組,程式配置又是另外一個組等等。當需要重建列表頁的時候,隻需要刪除列表頁這個組裡面全部的數據,而不會影響到別的組的數據。
 
測試瞭幾種方案,還是以下的方案最為理想和高速,我們先看代碼,再說原理:
 
 
<?php
class MyCache
{
    private $mmc = null;
    private $group = null;
    private $version = 1;
    function __construct($group){
        if(!class_exists('mmcache')){
            $this->mmc = false;
            return;
        }
        $this->mmc = new memcache();
        $this->mmc->addServer('192.168.1.5', 11211);
        $this->mmc->addServer('192.168.1.6', 11211);
        $this->group = $group;
        $this->version = $this->mmc->get('version_'.$group);
    }
    function set($key, $var, $expire=3600){
        if(!$this->mmc)return;
        return $this->mmc->set($this->group.'_'.$this->version.'_'.$key, $var, $expire);
    }
    function get($key){
        if(!$this->mmc)return;
        return $this->mmc->get($this->group.'_'.$this->version.'_'.$key);
    }
    function incr($key, $value=1){
        if(!$this->mmc)return;
        return $this->mmc->increment($this->group.'_'.$this->version.'_'.$key, $value);
    }
    function decr($key, $value=1){
        if(!$this->mmc)return;
        return $this->mmc->decrement($this->group.'_'.$this->version.'_'.$key, $value);
    }
    function delete($key){
        if(!$this->mmc)return;
        return $this->mmc->delete($this->group.'_'.$this->version.'_'.$key);
    }
    function flush(){
        if(!$this->mmc)return;
        ++$this->version;
        $this->mmc->set('version_'.$this->group, $this->version);
    }
}
?>
 
上面的類比較完整,包括鏈接Memcache服務,設置和獲取值,增減值,還有刪除key和全刪除(flush)。這裡包括瞭常規的Memcache操作功能,和對全刪除(flush)操作的擴展。
 
從代碼可以看到,支持組的flush功能的實現,是通過version這個key來實現的,也就是每次存該組的變量的時候,變量的key都會加入version值,version值是一個數字(從1開始),當存和取key的時候,version值都會被使用到。
 
當開發者要flush當前組的數據的時候,flush操作隻是簡單地改變一些version的值(加一),那麼,下次存取key的時候,將獲取不到原來的值——因為version改變瞭,也就是取的key名稱已經改變瞭。這樣原有的值會被Memcache自動回收,不會出現任何的效率開銷。而且程式上隻是增加一個version的存和取,數據量極小,對系統效率基本沒有任何影響。
 
通過以上的類,可以針對Memcache緩存進行組的操作,而這個PHP類,還可以繼續擴展,如加入socket直接訪問memcache的接口功能,這樣PHP環境中就不需要安裝memcache擴展類瞭,這樣更有效避免flush的誤操作瞭,而且在加入apc等緩存機制後,socket訪問memcache接口也不會比擴展慢多少。
 
另外,MyCache類還有個附加的功能:當memcache服務失效的時候,MyCache類隻是簡單返回空值,而不會直接出錯。
 
以下附帶MyCache類的使用方法:
 
 
// 引入定義
include('MyCache.php');
 
// 實例化
$mc = new MyCache('abc'); // 要有域
 
// 設置值
$mc->set('word', 'hello world', 900);
 
// 取得值
echo $mc->get('word');
 
// 刪除值
$mc->delete('word');
echo $mc->get('word');
 
$mc->set('counter', 1, 290000);
echo $mc->get('counter');
 
// 增加值
$mc->incr('counter');
$mc->incr('counter');
echo $mc->get('counter');Memcache是PHP開發中較常用到的緩存方法,在高並發的系統中是必不可少的組成部分。
在實際開發中,Memcache存在一個比較不盡人意的問題,就是Memcache不能支持對key進行的組操作。
 
組操作,也可以稱為域操作,比如說某個文章系統,在前臺部分使用Memcache緩存瞭列表頁數據、文章詳細頁數據,兩種數據的量都比較多。那麼,當後臺發佈瞭一篇文章的時候,列表頁就應該需要更新到最新的列表——可能涉及到許多個列表頁。當然,對文章詳細頁來說,它是不需要更新的。
 
好的,這個時候我們就需要刪除原有緩存,讓程式可以自動更新列表頁數據。但是使用Memcache的flush函數有個問題,就是它會清空全部的數據,包括列表頁和文章頁的數據,在大並發的條件下,全部緩存刪除後重建緩存的時候,將會有非常高的負載產生。
 
另外,還會有情況就是有些你不願意刪除的緩存變量,也會丟失瞭,比如說程式的配置,數據庫為瞭提速而存到緩存的表結構等。
 
所以我們需要一個支持組操作的緩存機制,我們就可以把列表頁設置成一個組,文章頁數據是另外一個組,程式配置又是另外一個組等等。當需要重建列表頁的時候,隻需要刪除列表頁這個組裡面全部的數據,而不會影響到別的組的數據。
 
測試瞭幾種方案,還是以下的方案最為理想和高速,我們先看代碼,再說原理:
 
 
<?php
class MyCache
{
    private $mmc = null;
    private $group = null;
    private $version = 1;
    function __construct($group){
        if(!class_exists('mmcache')){
            $this->mmc = false;
            return;
        }
        $this->mmc = new memcache();
        $this->mmc->addServer('192.168.1.5', 11211);
        $this->mmc->addServer('192.168.1.6', 11211);
        $this->group = $group;
        $this->version = $this->mmc->get('version_'.$group);
    }
    function set($key, $var, $expire=3600){
        if(!$this->mmc)return;
        return $this->mmc->set($this->group.'_'.$this->version.'_'.$key, $var, $expire);
    }
    function get($key){
        if(!$this->mmc)return;
        return $this->mmc->get($this->group.'_'.$this->version.'_'.$key);
    }
    function incr($key, $value=1){
        if(!$this->mmc)return;
        return $this->mmc->increment($this->group.'_'.$this->version.'_'.$key, $value);
    }
    function decr($key, $value=1){
        if(!$this->mmc)return;
        return $this->mmc->decrement($this->group.'_'.$this->version.'_'.$key, $value);
    }
    function delete($key){
        if(!$this->mmc)return;
        return $this->mmc->delete($this->group.'_'.$this->version.'_'.$key);
    }
    function flush(){
        if(!$this->mmc)return;
        ++$this->version;
        $this->mmc->set('version_'.$this->group, $this->version);
    }
}
?>
 
上面的類比較完整,包括鏈接Memcache服務,設置和獲取值,增減值,還有刪除key和全刪除(flush)。這裡包括瞭常規的Memcache操作功能,和對全刪除(flush)操作的擴展。
 
從代碼可以看到,支持組的flush功能的實現,是通過version這個key來實現的,也就是每次存該組的變量的時候,變量的key都會加入version值,version值是一個數字(從1開始),當存和取key的時候,version值都會被使用到。
 
當開發者要flush當前組的數據的時候,flush操作隻是簡單地改變一些version的值(加一),那麼,下次存取key的時候,將獲取不到原來的值——因為version改變瞭,也就是取的key名稱已經改變瞭。這樣原有的值會被Memcache自動回收,不會出現任何的效率開銷。而且程式上隻是增加一個version的存和取,數據量極小,對系統效率基本沒有任何影響。
 
通過以上的類,可以針對Memcache緩存進行組的操作,而這個PHP類,還可以繼續擴展,如加入socket直接訪問memcache的接口功能,這樣PHP環境中就不需要安裝memcache擴展類瞭,這樣更有效避免flush的誤操作瞭,而且在加入apc等緩存機制後,socket訪問memcache接口也不會比擴展慢多少。
 
另外,MyCache類還有個附加的功能:當memcache服務失效的時候,MyCache類隻是簡單返回空值,而不會直接出錯。
 
以下附帶MyCache類的使用方法:
 
 
// 引入定義
include('MyCache.php');
 
// 實例化
$mc = new MyCache('abc'); // 要有域
 
// 設置值
$mc->set('word', 'hello world', 900);
 
// 取得值
echo $mc->get('word');
 
// 刪除值
$mc->delete('word');
echo $mc->get('word');
 
$mc->set('counter', 1, 290000);
echo $mc->get('counter');
 
// 增加值
$mc->incr('counter');
$mc->incr('counter');
echo $mc->get('counter');
 
// 減少值
$mc->decr('counter');
echo $mc->get('counter');
 
// 按組刪
$mc->flush(); 
本文出自 “振中的技術記事本” 博客
 
// 減少值
$mc->decr('counter');
echo $mc->get('counter');
 
// 按組刪
$mc->flush(); 
作者“振中的技術記事本”

發佈留言

發佈留言必須填寫的電子郵件地址不會公開。 必填欄位標示為 *