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();
作者“振中的技術記事本”