緩存在實際使用當中應用很廣泛,可以減輕對服務器數據庫的訪問,提高運行速度。目前很多CMS內容管理系統中頻繁使用緩存機制來提高系統運行的效率。下面是一個寫得不錯的緩存類,可以參考下緩存的機制與寫法。
cache.php 代碼如下:
/* 用戶需要事先定義的常量: _CachePath_ 模板緩存路徑 _CacheEnable_ 自動緩存機制是否開啟,未定義或為空,表示關閉自動緩存機制 _ReCacheTime_ 自動重新緩存間隔時間,單位為秒,未定義或為空,表示關閉自動重新緩存 */ class cache { var $cachefile; var $cachefilevar; function cache() { //生成當前頁的Cache組文件名 $this->cachefilevar 及文件名 $this->cachefile //動態頁的參數不同對應的Cache文件也不同,但是每一個動態頁的所有Cache文件都有相同的文件名,只是擴展名不同 $s=array(".","/");$r=array("_",""); $this->cachefilevar=str_replace($s,$r,$_SERVER["SCRIPT_NAME"])."_".$_GET[_ActionVar_]; $this->cachefile=$this->cachefilevar.".".md5($_SERVER["REQUEST_URI"]); } //刪除當前頁/模塊的緩存 function delete() { //刪除當前頁的緩存 $d = dir(_CachePath_); $strlen=strlen($this->cachefilevar); //返回當前頁的所有Cache文件組 while (false !== ($entry = $d->read())) { if (substr($entry,0,$strlen)==$this->cachefilevar) { if (!unlink(_CachePath_."/".$entry)) {echo "Cache目錄無法寫入";exit;} } } } //判斷是否已Cache過,以及是否需要Cache function check() { //如果設置了緩存更新間隔時間 _ReCacheTime_ if (_ReCacheTime_+0>0) { //返回當前頁Cache的最後更新時間 $var=@file(_CachePath_."/".$this->cachefilevar);$var=$var[0]; //如果更新時間超出更新間隔時間則刪除Cache文件 if (time()-$var>_ReCacheTime_) { $this->delete();$ischage=true; } } //返回當前頁的Cache $file=_CachePath_."/".$this->cachefile; //判斷當前頁Cache是否存在 且 Cache功能是否開啟 return (file_exists($file) and _CacheEnable_ and !$ischange); } //讀取Cache function read() { //返回當前頁的Cache $file=_CachePath_."/".$this->cachefile; //讀取Cache文件的內容 if (_CacheEnable_) return readfile($file); else return false; } //生成Cache function write($output) { //返回當前頁的Cache $file=_CachePath_."/".$this->cachefile; //如果Cache功能開啟 if (_CacheEnable_) { //把輸出的內容寫入Cache文件 $fp=@fopen($file,'w'); if (!@fputs($fp,$output)) {echo "模板Cache寫入失敗";exit;} @fclose($fp); //如果設置了緩存更新間隔時間 _ReCacheTime_ if (_ReCacheTime_+0>0) { //更新當前頁Cache的最後更新時間 $file=_CachePath_."/".$this->cachefilevar; $fp=@fopen($file,'w'); if (!@fwrite($fp,time())) {echo "Cache目錄無法寫入";exit;} @fclose($fp); } } } } ?>
類的使用:
check()) { $template=$cache->read(); } else { ob_start(); ob_implicit_flush(0); ?> 頁面內容。。。。 write($template); } ?>
有些信息比方經常不變的,但是還是能變的信息放在緩存中以加快顯示速度,這是很有價值的,所謂的緩存,通俗的理解就是一些保存在服務器端的共用信息。它是於服務器同生死的,我們在保存緩存的時候可以指定下次更新的時間的判斷,比方要在5分鐘更新一次,可以記錄上次更新的時間,和當前時間比較,如果大於 5 分鐘 ,讀取數據庫,更新換成,否則直接讀取緩存數據,當然,緩存需要客戶端用戶激活的,只需一次。
ob_start()函數:打開輸出緩沖區。
函數格式 void ob_start(void)
說明:當緩沖區激活時,所有來自PHP程序的非文件頭信息均不會發送,而是保存在內部緩沖區。為了輸出緩沖區的內容,可以使用ob_end_flush()或flush()輸出緩沖區的內容。
函數格式:flush()
說明:這個函數經常使用,效率很高。
ob_get_contents :返回內部緩沖區的內容。
函數格式:string ob_get_contents(void)
說明:這個函數會返回當前緩沖區中的內容,如果輸出緩沖區沒有激活,則返回 FALSE。
ob_get_length:返回內部緩沖區的長度。
函數格式:int ob_get_length(void)
說明:這個函數會返回當前緩沖區中的長度;和ob_get_contents一樣,如果輸出緩沖區沒有激活,則返回 FALSE。
ob_end_clean:刪除內部緩沖區的內容,並且關閉內部緩沖區。
函數格式:void ob_end_clean(void)
說明:這個函數不會輸出內部緩沖區的內容而是把它刪除。
ob_end_flush:發送內部緩沖區的內容到浏覽器,並且關閉輸出緩沖區
函數格式:void ob_end_flush(void)
說明:這個函數發送輸出緩沖區的內容(如果有的話)。
函數格式:void ob_implicit_flush ([int flag])
說明:默認為關閉緩沖區,打開絕對輸出後,每個腳本輸出都直接發送到浏覽器,不再需要調用 flush()。
如果你的網站MySQL數據庫的速度比較慢,你就需要看重網站的緩存了。用過 WordPress 的朋友都知道,它有一個插件叫 WP Super Cache, 可以將 WordPress 的頁面在第一次生成時儲存成靜態頁面,當再次請求這個頁面時,就省去了讀取數據庫的時間。這裡討論的就是這種技術。
第一個問題就是怎樣獲取 PHP 輸出的內容了。獲取輸出內容的原因很簡單,因為我們可以把輸出的內容儲存起來,當訪客再次光臨的時候就把事先存好的內容給他。
實現這些目的其實也同樣簡單。我們只要把函數 ob_start() 在內容輸出前調用,然後在所有內容輸出完成後調用 ob_get_contents() 獲取輸出的內容,再在此後調用 ob_end_flush() 表示結束就可以了,一個簡單的例子如下:
在 PHP 標簽之外的輸出可以被記錄。
我被記錄了。' ?>
程序運行結果:
在 PHP 標簽之外的輸出可以被記錄。 我被記錄了。 在 PHP 標簽之外的輸出可以被記錄。 我被記錄了。
可見,$cache變量保存了之前的輸出結果。就是說,我們可以通過cache來減少PHP的結果輸出。
有時我們有這樣的習慣,對於管理員是不啟用緩存的,而對游客則啟用緩存。這個時候,其實實現起來也比較簡單。我們可以自己編寫兩個函數 cache($id) 和 end_cache($id),分別表示緩存開始和緩存結束,然後代碼如下(這裡有三個函數省略了):
有的時候,站點可能會根據需要,建立了專門為移動設備設計的頁面。那麼,這種情況下我們就應該將 $id 擴展一下。這種擴展有很多種方法,比如添加另一個參數,將移動設備的頁面存在不同於桌面設備的文件夾中,而這些頁面使用相同的 $id . 另外還有一種做法,就是將原來的 $id 與移動設備的 User-agent 糅合在一起,md5() 一下就可以了。我偏向於前面那種做法。當然肯定還有其它類似的做法,總之中心思想就是把緩存的標記 ($id) 設置成不一樣的東西,並且當用戶回來後還能區別得出它們,就可以了。
還有的時候,一個網站有多種用戶角色,可能要給相應的用戶相應的緩存。當然,只需遵循上面的原則。
ob_start() 和 ob_end_flush() 是遞歸處理的。也就是說,可以在調用 ob_end_flush() 之前,調用若干次 ob_start() . 例如:
有的時候,站點可能會根據需要,建立了專門為移動設備設計的頁面。那麼,這種情況下我們就應該將 $id 擴展一下。這種擴展有很多種方法,比如添加另一個參數,將移動設備的頁面存在不同於桌面設備的文件夾中,而這些頁面使用相同的 $id . 另外還有一種做法,就是將原來的 $id 與移動設備的 User-agent 糅合在一起,md5() 一下就可以了。我偏向於前面那種做法。當然肯定還有其它類似的做法,總之中心思想就是把緩存的標記 ($id) 設置成不一樣的東西,並且當用戶回來後還能區別得出它們,就可以了。
還有的時候,一個網站有多種用戶角色,可能要給相應的用戶相應的緩存。當然,只需遵循上面的原則。
ob_start() 和 ob_end_flush() 是遞歸處理的。也就是說,可以在調用 ob_end_flush() 之前,調用若干次 ob_start() . 例如:
'; ob_start(); echo 'content2'.'
'; $output1 = ob_get_contents(); echo $output1.'
'; ob_end_flush(); $output2 = ob_get_contents(); echo $output2.'
'; ob_end_flush(); ?>
程序運行結果:
content1 content2 content2 content1 content2 content2