Memcache是PHP開發中較常用到的緩衝方法,在高並發的系統中是必不可少的組成部分。
在實際開發中,Memcache存在一個比較不盡人意的問題,就是Memcache不能支援對key進行的組操作。
組操作,也可以稱為網域作業,比如說某個文章系統,在前台部分使用Memcache緩衝了列表頁資料、文章詳細頁資料,兩種資料的量都比較多。那麼,當後台發布了一篇文章的時候,列表頁就應該需要更新到最新的列表——可能涉及到許多個列表頁。當然,對文章詳細頁來說,它是不需要更新的。
好的,這個時候我們就需要刪除原有緩衝,讓程式可以自動更新列表頁資料。但是使用Memcache的flush函數有個問題,就是它會清空全部的資料,包括列表頁和文章頁的資料,在大並發的條件下,全部緩衝刪除後重建緩衝的時候,將會有非常高的負載產生。
另外,還會有情況就是有些你不願意刪除的緩衝變數,也會丟失了,比如說程式的配置,資料庫為了提速而存到緩衝的表結構等。
所以我們需要一個支援組操作的緩衝機制,我們就可以把列表頁設定成一個組,文章頁資料是另外一個組,程式配置又是另外一個組等等。當需要重建列表頁的時候,只需要刪除列表頁這個組裡面全部的資料,而不會影響到別的組的資料。
測試了幾種方案,還是以下的方案最為理想和高速,我們先看代碼,再說原理:
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函數有個問題,就是它會清空全部的資料,包括列表頁和文章頁的資料,在大並發的條件下,全部緩衝刪除後重建緩衝的時候,將會有非常高的負載產生。
另外,還會有情況就是有些你不願意刪除的緩衝變數,也會丟失了,比如說程式的配置,資料庫為了提速而存到緩衝的表結構等。
所以我們需要一個支援組操作的緩衝機制,我們就可以把列表頁設定成一個組,文章頁資料是另外一個組,程式配置又是另外一個組等等。當需要重建列表頁的時候,只需要刪除列表頁這個組裡面全部的資料,而不會影響到別的組的資料。
測試了幾種方案,還是以下的方案最為理想和高速,我們先看代碼,再說原理:
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();
作者“振中的技術記事本”
http://www.bkjia.com/PHPjc/478661.htmlwww.bkjia.comtruehttp://www.bkjia.com/PHPjc/478661.htmlTechArticleMemcache是PHP開發中較常用到的緩衝方法,在高並發的系統中是必不可少的組成部分。 在實際開發中,Memcache存在一個比較不盡人意的問題,...