一個不錯的PHP緩存類介紹和PHP緩存函數介紹以及使用

jopen 10年前發布 | 35K 次閱讀 PHP PHP開發

緩存在實際使用當中應用很廣泛,可以減輕對服務器數據庫的訪問,提高運行速度。目前很多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);    
            }    
        }    
    }    
}    
?>  </pre><p> </p>

類的使用:

    <?php    
        define("_CachePath_","./cache/");    
        define("_CacheEnable_","1");    
        define("_ReCacheTime_","43200");    
        include('cache.php');    
        $cache=new cache();    
        if ($cache->check())   
        {    
            $template=$cache->read();    
        }  
        else   
        {    
            ob_start();    
            ob_implicit_flush(0);    
    ?>    
        頁面內容。。。。    
    <?php    
            $template = ob_get_contents();    
            $cache->write($template);    
        }    
    ?>    


 

 

 

PHP的緩存相關函數介紹

 

有些信息比方經常不變的,但是還是能變的信息放在緩存中以加快顯示速度,這是很有價值的,所謂的緩存,通俗的理解就是一些保存在服務器端的共用信息。它是 于服務器同生死的,我們在保存緩存的時候可以指定下次更新的時間的判斷,比方要在5分鐘更新一次,可以記錄上次更新的時間,和當前時間比較,如果大于 5 分鐘 ,讀取數據庫,更新換成,否則直接讀取緩存數據,當然,緩存需要客戶端用戶激活的,只需一次。

ob_start()函數

ob_start()函數:打開輸出緩沖區。

函數格式 void ob_start(void)

說明:當緩沖區激活時,所有來自PHP程序的非文件頭信息均不會發送,而是保存在內部緩沖區。為了輸出緩沖區的內容,可以使用ob_end_flush()或flush()輸出緩沖區的內容。

flush()函數

函數格式:flush()

說明:這個函數經常使用,效率很高。

string ob_get_contents()函數

ob_get_contents :返回內部緩沖區的內容。

函數格式:string ob_get_contents(void)

說明:這個函數會返回當前緩沖區中的內容,如果輸出緩沖區沒有激活,則返回 FALSE。

ob_get_length()函數

ob_get_length:返回內部緩沖區的長度。

函數格式:int ob_get_length(void)

說明:這個函數會返回當前緩沖區中的長度;和ob_get_contents一樣,如果輸出緩沖區沒有激活,則返回 FALSE。

ob_end_clean()函數

ob_end_clean:刪除內部緩沖區的內容,并且關閉內部緩沖區。

函數格式:void ob_end_clean(void)

說明:這個函數不會輸出內部緩沖區的內容而是把它刪除。

ob_end_flush()函數

ob_end_flush:發送內部緩沖區的內容到瀏覽器,并且關閉輸出緩沖區

函數格式:void ob_end_flush(void)

說明:這個函數發送輸出緩沖區的內容(如果有的話)。

ob_implicit_flush()函數

函數格式:void ob_implicit_flush ([int flag])

說明:默認為關閉緩沖區,打開絕對輸出后,每個腳本輸出都直接發送到瀏覽器,不再需要調用 flush()。

PHP對靜態頁面的緩存技術研究

如果你的網站MySQL數據庫的速度比較慢,你就需要看重網站的緩存了。用過 WordPress 的朋友都知道,它有一個插件叫 WP Super Cache, 可以將 WordPress 的頁面在第一次生成時儲存成靜態頁面,當再次請求這個頁面時,就省去了讀取數據庫的時間。這里討論的就是這種技術。

第一個問題就是怎樣獲取 PHP 輸出的內容了。獲取輸出內容的原因很簡單,因為我們可以把輸出的內容儲存起來,當訪客再次光臨的時候就把事先存好的內容給他。

實現這些目的其實也同樣簡單。我們只要把函數 ob_start() 在內容輸出前調用,然后在所有內容輸出完成后調用 ob_get_contents() 獲取輸出的內容,再在此后調用 ob_end_flush() 表示結束就可以了,一個簡單的例子如下:

    <?php ob_start(); ?>  

    <p>在 PHP 標簽之外的輸出可以被記錄。</p>  
    <?php echo '<p>我被記錄了。</p>' ?>  

    <?php $cache = ob_get_contents(); ?>  
    <?php  
        /* 在這里添加任何處理 $cache 的代碼 */  
        echo $cache;  
    ?>  

    <?php ob_end_flush(); ?>  


程序運行結果:

 
在 PHP 標簽之外的輸出可以被記錄。  
我被記錄了。  

在 PHP 標簽之外的輸出可以被記錄。  
我被記錄了。 

      可見,$cache變量保存了之前的輸出結果。就是說,我們可以通過cache來減少PHP的結果輸出。

有時我們有這樣的習慣,對于管理員是不啟用緩存的,而對游客則啟用緩存。這個時候,其實實現起來也比較簡單。我們可以自己編寫兩個函數 cache($id) 和 end_cache($id),分別表示緩存開始和緩存結束,然后代碼如下(這里有三個函數省略了):

    <?php  

    function is_admin() {  
      /* 該函數用于測試當前用戶是否是管理員,若是管理員則返回 true. */  
    }  

    function show_cache($id) {  
      /* 根據 $id 讀取并顯示緩存內容,若無緩存則返回 false. */  
    }  

    function save_cache($id, $content) {  
      /* 將標識符為 $id 的緩存,以內容 $content 儲存。 */  
    }  

    function cache($id) {  
      if (is_admin())  
        return false;  
      if (show_cache($id))  
        return false;  
      ob_start();  
      return true;  
    }  

    function end_cache($id) {  
      if (is_admin())  
        return false;  
      save_cache($id, ob_get_contents());  
      ob_end_flush();  
      return true;  
    }  

    ?>  

有的時候,站點可能會根據需要,建立了專門為移動設備設計的頁面。那么,這種情況下我們就應該將 $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() . 例如:

    <?php  
    ob_start();  
    echo 'content1'.'<br />';  

    ob_start();  
    echo 'content2'.'<br />';  

    $output1 = ob_get_contents();   
    echo $output1.'<br />';  
    ob_end_flush();  
    $output2 = ob_get_contents();   
    echo $output2.'<br />';  
    ob_end_flush();  
    ?>  

程序運行結果:

content1  
content2  
content2  

content1  
content2  
content2  
 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!