Memcache是PHP開發中較常用到的快取方法,在高並發的系統中是不可或缺的組成部分。
在實際開發中,Memcache存在一個比較不盡人意的問題,就是Memcache不能支援對key進行的群組操作。
群組操作,也可以稱為網域操作,比如說某個文章系統,在前台部分使用Memcache快取了列表頁數據、文章詳細頁數據,兩種數據的量都比較多。那麼,當後台發布了一篇文章的時候,清單頁就應該需要更新到最新的清單——可能涉及許多清單頁面。當然,對文章詳細頁來說,它是不需要更新的。
好的,這個時候我們就需要刪除原有緩存,讓程式可以自動更新清單頁資料。但是使用Memcache的flush函數有個問題,就是它會清空全部的數據,包括列表頁和文章頁的數據,在大並發的條件下,全部緩存刪除後重建緩存的時候,將會有非常高的負載產生。
另外,還會有情況就是有些你不願意刪除的快取變量,也會遺失了,比如說程式的配置,資料庫為了提速而存到快取的表結構等。
所以我們需要一個支援組操作的快取機制,我們就可以把清單頁設定成一個群組,文章頁資料是另一個群組,程式配置又是另一個群組等等。當需要重建列表頁的時候,只需要刪除列表頁這個組裡面全部的數據,而不會影響到別的組的數據。
測試了幾種方案,還是以下的方案最為理想和高速,我們先看程式碼,再說原理:
<?php class MyCache { private $mmc = null; private $group = null; 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); if(empty($this->version)){ $this->version=1; } } function set($key, $var, $expire=3600){ if(!$this->mmc)return; $this->check_version($this->group,$this->version,$key); return $this->mmc->set($this->group.'_'.$this->version.'_'.$key, $var, $expire); } function get($key){ if(!$this->mmc)return; $this->check_version($this->group,$this->version,$key); 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); } function check_version($goup,$version,$key){ if($version>1){ $version_old=$version-1; return $this->mmc->delete($goup.'_'.$version_old.'_'.$key); } } } ?>
上面的類別比較完整,包括連結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();
以上是詳細介紹PHP支援群組操作的Memcache類別(網域操作)的程式碼實例的內容,而更多相關內容請關注PHP中文網(www.php.cn)!