PHP,一門最近幾年興起的web設(shè)計腳本語言,由于它的強大和可伸縮性,近幾年來得到長足的發(fā)展,php相比傳統(tǒng)的asp網(wǎng)站,在速度上有絕對的優(yōu)勢,想mssql轉(zhuǎn)6萬條數(shù)據(jù)php如需要40秒,asp不下2分鐘.但是,由于網(wǎng)站的數(shù)據(jù)越來越多,我們渴求能更快速的調(diào)用數(shù)據(jù),不必要每次都從數(shù)據(jù)庫掉,我們可以從其他的地方,比方一個文件,或者某個內(nèi)存地址,這就是php的緩存技術(shù),也就是Cache技術(shù).
一般來說,緩存的目的是把數(shù)據(jù)放在一個地方讓訪問的更快點,毫無疑問,內(nèi)存是最快的,但是,幾百M的數(shù)據(jù)能往內(nèi)存放么?這不現(xiàn)實,當(dāng)然,有的時候臨時放如服務(wù)器緩存,如ob_start()這個緩存頁面開啟的話在發(fā)送文件頭之前頁面內(nèi)容都被緩存在內(nèi)存中,知道等頁面輸出自動清楚或者等待ob_get_contents的返回,[或者被ob_end_clean顯示的清除,這在靜態(tài)頁面的生成中能很好的利用,在模板中能得到很好的體現(xiàn),我的這篇文章深入的討論了:
談PHP生成靜態(tài)頁面,這是一種方式,但這是臨時性的,不是解決我們問題的好方法.
另外,在asp中有一對象application,可以保存公用的參數(shù),這也算點緩存,但在php,我至今沒看到開發(fā)者產(chǎn)出這種對象,的確,沒必要.asp.net的頁面緩存技術(shù)就用的是viewstate,而cache就是文件關(guān)聯(lián),(不一定準(zhǔn)確),文件被修改,更新緩存,文件沒被修改而且不超時(注釋1),就讀取緩存,返回結(jié)果,就是這個思路,看看這個源碼:
<?php
class cache{
/*
Class Name: cache
Description: control to cache data,cache_out_time is a array to save cache date time out.
Version: 1.0
Author: 老農(nóng) cjjer
Last modify:2006-2-26
Author URL: http://www.cjjer.com
*/
private cache_dir;
private expireTime=180;//緩存的時間是 60 秒
function __construct(cache_dirname){
if(!@is_dir(cache_dirname)){
if(!@mkdir(cache_dirname,0777)){
this->warn('緩存文件不存在而且不能創(chuàng)建,需要手動創(chuàng)建.');
return false;
}
}
this->cache_dir = cache_dirname;
}
function __destruct(){
echo 'Cache class bye.';
}
function get_url() {
if (!isset(_SERVER['REQUEST_URI'])) {
url = _SERVER['REQUEST_URI'];
}else{
url = _SERVER['SCRIPT_NAME'];
url .= (!empty(_SERVER['QUERY_STRING'])) ? '?' . _SERVER['QUERY_STRING'] : '';
}
return url;
}
function warn(errorstring){
echo "<b><font color='red'>發(fā)生錯誤:<pre>".errorstring."</pre></font></b>";
}
function cache_page(pageurl,pagedata){
if(!fso=fopen(pageurl,'w')){
this->warns('無法打開緩存文件.');//trigger_error
return false;
}
if(!flock(fso,LOCK_EX)){//LOCK_NB,排它型鎖定
this->warns('無法鎖定緩存文件.');//trigger_error
return false;
}
if(!fwrite(fso,pagedata)){//寫入字節(jié)流,serialize寫入其他格式
this->warns('無法寫入緩存文件.');//trigger_error
return false;
}
flock(fso,LOCK_UN);//釋放鎖定
fclose(fso);
return true;
}
function display_cache(cacheFile){
if(!file_exists(cacheFile)){
this->warn('無法讀取緩存文件.');//trigger_error
return false;
}
echo '讀取緩存文件:'.cacheFile;
//return unserialize(file_get_contents(cacheFile));
fso = fopen(cacheFile, 'r');
data = fread(fso, filesize(cacheFile));
fclose(fso);
return data;
}
function readData(cacheFile='default_cache.txt'){
cacheFile = this->cache_dir."/".cacheFile;
if(file_exists(cacheFile)&&filemtime(cacheFile)>(time()-this->expireTime)){
data=this->display_cache(cacheFile);
}else{
data="from here wo can get it from mysql database,update time is <b>".date('l dS /of F Y h:i:s A')."</b>,過期時間是:".date('l dS /of F Y h:i:s A',time()+this->expireTime)."----------";
this->cache_page(cacheFile,data);
}
return data;
}
}
?>
下面我打斷這個代碼逐行解釋.
三:程序透析
這個緩存類(類沒什么好怕的.請繼續(xù)看)名稱是cache,有2個屬性:
private cache_dir;
private expireTime=180;
cache_dir是緩存文件所放的相對網(wǎng)站目錄的父目錄, expireTime(注釋一)是我們緩存的數(shù)據(jù)過期的時間,主要是這個思路:
當(dāng)數(shù)據(jù)或者文件被加載的時候,先判斷緩存文件存在不,返回false ,文件最后修改時間和緩存的時間和比當(dāng)前時間大不,大的話說明緩存還沒到期,小的話返回false,當(dāng)返回false的時候,讀取原始數(shù)據(jù),寫入緩存文件中,返回數(shù)據(jù).,
接著看程序:
function __construct(cache_dirname){
if(!@is_dir(cache_dirname)){
if(!@mkdir(cache_dirname,0777)){
this->warn('緩存文件不存在而且不能創(chuàng)建,需要手動創(chuàng)建.');
return false;
}
}
this->cache_dir = cache_dirname;
}
當(dāng)類第一次被實例的時候構(gòu)造默認(rèn)函數(shù)帶參數(shù)緩存文件名稱,如文件不存在,創(chuàng)建一個有編輯權(quán)限的文件夾,創(chuàng)建失敗的時候拋出異常.然后把cache類的 cache_dir屬性設(shè)置為這個文件夾名稱,我們的所有緩存文件都是在這個文件夾下面的.
function __destruct(){
echo 'Cache class bye.';
}
這是class類的析構(gòu)函數(shù),為了演示,我們輸出一個字符串表示我們釋放cache類資源成功.
function warn(errorstring){
echo "<b><font color='red'>發(fā)生錯誤:<pre>".errorstring."</pre></font></b>";
}
這個方法輸出錯誤信息.
function get_url() {
if (!isset(_SERVER['REQUEST_URI'])) {
url = _SERVER['REQUEST_URI'];
}else{
url = _SERVER['SCRIPT_NAME'];
url .= (!empty(_SERVER['QUERY_STRING'])) ? '?' . _SERVER['QUERY_STRING'] : '';
}
return url;
}
這個方法返回當(dāng)前url的信息,這是我看國外很多人的cms系統(tǒng)這樣做,主要是緩存x.php?page=1,x.php?page=2,等這種文件的,這里列出是為了擴展的這個cache類功能的.
function cache_page(pageurl,pagedata){
if(!fso=fopen(pageurl,'w')){
this->warns('無法打開緩存文件.');//trigger_error
return false;
}
if(!flock(fso,LOCK_EX)){//LOCK_NB,排它型鎖定
this->warns('無法鎖定緩存文件.');//trigger_error
return false;
}
if(!fwrite(fso,pagedata)){//寫入字節(jié)流,serialize寫入其他格式
this->warns('無法寫入緩存文件.');//trigger_error
return false;
}
flock(fso,LOCK_UN);//釋放鎖定
fclose(fso);
return true;
}
cache_page方法分別傳入的是緩存的文件名稱和數(shù)據(jù),這是把數(shù)據(jù)寫到文件里的方法,先用fopen打開文件,然后調(diào)用句柄鎖定這個文件,然后用fwrite寫入文件,最后釋放這個句柄,任何一步發(fā)生錯誤將拋出錯誤. 您可能看到 這個注釋
寫入字節(jié)流,serialize寫入其他格式
,順便一提的是如果我們要把一個數(shù)組,(可以從MySQL數(shù)據(jù)庫里面select查詢除了的結(jié)果)用serialize函數(shù)寫入,用unserialize讀取到原來的類型.
function display_cache(cacheFile){
if(!file_exists(cacheFile)){
this->warn('無法讀取緩存文件.');//trigger_error
return false;
}
echo '讀取緩存文件:'.cacheFile;
//return unserialize(file_get_contents(cacheFile));
fso = fopen(cacheFile, 'r');
data = fread(fso, filesize(cacheFile));
fclose(fso);
return data;
}
這是由文件名稱讀取緩存的方法,直接打開文件,讀取全部,如果文件不存在的或者無法讀取的話返回false,當(dāng)然,你感到不人性的話,可以重新生成緩存.
function readData(cacheFile='default_cache.txt'){
cacheFile = this->cache_dir."/".cacheFile;
if(file_exists(cacheFile)&&filemtime(cacheFile)>(time()-this->expireTime)){
data=this->display_cache(cacheFile);
}else{
data="from here wo can get it from mysql database,update time is <b>".date('l dS /of F Y h:i:s A')."</b>,過期時間是:".date('l dS /of F Y h:i:s A',time()+this->expireTime)."----------";
this->cache_page(cacheFile,data);
}
return data;
}
這個函數(shù)是我們調(diào)用的方法,可以寫成接口的方法,由傳入?yún)?shù)判斷文件存在不,文件最后修改時間+expireTime的時間是不是過了當(dāng)前時間(大于的話說明沒有過期),如果文件不存在或者已經(jīng)過期,重新加載原始數(shù)據(jù),這里,為了簡單期間,我們是直接源是字符串,您可以把cache類繼承某類,取到數(shù)據(jù)庫的數(shù)據(jù).(注釋2)
四:補充說明,結(jié)語
注釋一 :這個緩存的時間您可以自己調(diào),可以根據(jù)時間情況讀取數(shù)組,xml,緩存等,請按照您的方便,值得一提的是緩存的時間(也就是緩存的key)也用緩存控制,.這在cms系統(tǒng)中被廣泛使用,他們把要更新的key放在緩存中,非常容易控制全戰(zhàn).
注釋二: php5開始支持類繼承,這是讓人興奮的,把網(wǎng)站全局休息寫在一個配置的類里面,再寫與數(shù)據(jù)層交互的類(如與MySQL交互的類),我們的這個cache類繼承數(shù)據(jù)交互的類,可以非常容易的讀取數(shù)據(jù)庫,這是外話,此處不再展開,有時間和大家詳談.
特別說明,這個類文件針對的php5以上版本,其他版本的請不要使用類.