最近在看一個關(guān)于無限分類的緩存技術(shù)。
php+mysql緩存技術(shù)的實現(xiàn)
本教程適合于那些對緩存SQL查詢以減少數(shù)據(jù)庫連接與執(zhí)行的負(fù)載、提高腳本性能感興趣的PHP程序員。
概述許多站點使用數(shù)據(jù)庫作為站點數(shù)據(jù)存儲的容器。數(shù)據(jù)庫包含了產(chǎn)器信息、目錄結(jié)構(gòu)、文章或者留言本,有些數(shù)據(jù)很可能是完全靜態(tài)的,這些將會從一個緩存系統(tǒng)中得到的極大好處。
這樣一個系統(tǒng)通過把SQL查詢的結(jié)果緩存到系統(tǒng)的一個文件中存儲,從而阻止連接數(shù)據(jù)庫,構(gòu)造查詢與取得返回結(jié)果而提高了響應(yīng)時間。
有些系統(tǒng)數(shù)據(jù)庫并不是放在WEB服務(wù)器上的,這樣需要一個html' target='_blank'>遠(yuǎn)程連接(TCP或者其它類似的),或者從數(shù)據(jù)庫中獲取大量的數(shù)據(jù),這樣你得忍受更多時間,這決定于系統(tǒng)響應(yīng)時間與資源利用。
前提本教程使用MySQL作為數(shù)據(jù)庫。你需要安裝MySQL(www.mysql.com下載是有效的)和激活PHP MYSQL擴(kuò)展(默認(rèn)情況是激活的)。
由于要查詢數(shù)據(jù)庫,你需要知識一些SQL(結(jié)構(gòu)化查詢語言)的基本常識。 www.it165.net
緩存SQL查詢結(jié)果
為什么要緩存查詢結(jié)果?
緩存查詢結(jié)果能極大地改進(jìn)腳本執(zhí)行時間和資源需求。
緩存SQL查詢結(jié)果也允許你通過后期處理數(shù)據(jù)。如果你用文件緩存去存儲全部腳本的輸出結(jié)果(HTML輸出),這樣可能是行不通的。
當(dāng)你執(zhí)行一個SQL查詢時,點典的處理過程是:
l 連接數(shù)據(jù)庫
l 準(zhǔn)備SQL查詢
l 發(fā)送查詢到數(shù)據(jù)庫
l 取得返回結(jié)果
l 關(guān)閉數(shù)據(jù)庫連接
以上方法非常占用資源并且相反的影響了腳本的性能。只能通過取得的大量返回數(shù)據(jù)和數(shù)據(jù)庫服務(wù)器的位置這二個要素來相互協(xié)調(diào)。盡管持續(xù)連接可以改進(jìn)連接數(shù)據(jù)庫時的負(fù)載,但非常耗費內(nèi)存資源,如果獲取的是大量的數(shù)據(jù),那么存儲的全部時間會非常短暫。
創(chuàng)建一條SQL查詢:
SQL(結(jié)構(gòu)化查詢語言)查詢被用作操作數(shù)據(jù)庫及它內(nèi)容的接口。SQL可用于定義和編輯表的結(jié)構(gòu),插入數(shù)據(jù)到表,更新或刪除表中的信息。
SQL是用于與數(shù)據(jù)通訊的語言,在大多數(shù)PHP數(shù)據(jù)庫擴(kuò)展(MySQL,ODBC,Oracle等)通過傳遞SQL查詢到數(shù)據(jù)庫中來管理整個過程。
本教程中,僅僅用select語言來獲取數(shù)據(jù)庫中的數(shù)據(jù)。這些數(shù)據(jù)將被緩存,之后將用作數(shù)據(jù)源。
決定什么時候更新緩存:
根據(jù)程序的需要,緩存可以采取多種形式。最常見的3種方式是:
l 時間觸發(fā)緩存(過期的時間戳)
l 內(nèi)容改變觸發(fā)緩存(發(fā)現(xiàn)數(shù)據(jù)改變后,相應(yīng)地更新緩存)
l 人工觸發(fā)緩存(人工的方式告知系統(tǒng)信息超期并且強(qiáng)制產(chǎn)生新的緩存)
你的緩存需求可能是以上原理的一個或多個的綜合。本教程將討論時間觸發(fā)方式。然而,在一個全面的緩存機(jī)制中,3種方式的綜合將被使用。
緩存結(jié)果:
基本的緩存是用PHP的兩個函數(shù)serialize()和unserialize()(譯注:這二個函數(shù)分別代表序列化與反序列化)。
函數(shù)serialize()用于存儲PHP的值,它能保證不失去這些值的類型和結(jié)構(gòu)。
事實上,PHP的session擴(kuò)展是用序列化過的變量,把session變量($_SESSION)存儲在系統(tǒng)的一個文件中。
函數(shù)unserialize()與以上操作相反并且使序列化過的字符串返回到它原來的結(jié)構(gòu)和數(shù)據(jù)內(nèi)容。
在本例中,以一個電子商務(wù)商店為例。商店有2個基本表,categories和products(此處為原始數(shù)據(jù)庫表名).product表可能每天都在變化,categories仍然是不變靜止的。
要顯示產(chǎn)品,你可以用一個輸出緩存腳本來存儲輸出的HTML結(jié)果到一個文件中。然而categories表可能需要后期處理。例如,所有的目錄通過變量 category_id(通過$_REQUEST['category_id']來取得)被顯示,你可能希望高亮當(dāng)前被選擇的目錄。表categories結(jié)構(gòu)
Field Type Key Extra
category_id int(10) unsigned PRI auto_incremen
category_name varchar(255)
category_description text
在本例中,通過時間觸發(fā)緩存技術(shù)被運用,設(shè)定一段時間后讓其緩存SQL輸出過期。在此特殊的例子中,定一段時間為24小時。
序列化例子:
l 連接數(shù)據(jù)庫
l 執(zhí)行查詢
l 取得所有結(jié)果構(gòu)成一個數(shù)組以便后面你可以訪問
l 序列化數(shù)組
l 保存序列化過的數(shù)組到文件中*/
$file = ‘sql_cache.txt’;
$link = mysql_connect(‘localhost’,'username’,'password’) or die (mysql_error());
mysql_select_db(‘shop’) or die (mysql_error());
/* 構(gòu)造SQL查詢 */
$query = “SELECT * FROM categories”;
$result = mysql_query($query) or die (mysql_error());
while ($record = mysql_fetch_array($result) ){
$records[] = $record;
}
$OUTPUT = serialize($records);
$fp = fopen($file,”w”); // 以寫權(quán)限的方式打開文件
fputs($fp, $OUTPUT);
//fwrite($fp,$OUTPUT);
fclose($fp);
/*查看sql_cache.txt文件,里面的內(nèi)容可能類似這樣的:
a:1:{i:0;a:6:{i:0;s:1:”1″;s:11:”category_id”;s:1:”1″;i:1;s:9:”Computers”;s:13:”category_name”;s:9:”Computers” ;i:2;s:25:”Description for computers”;s:20:”category_description”;s:25:”Description for computers”;}}
這個輸出是它的變量和類型的內(nèi)部表現(xiàn)形式。假若你用mysql_fetch_array()函數(shù)返回數(shù)字索引的數(shù)組和一個關(guān)聯(lián)的數(shù)組(這就是為什么數(shù)據(jù)看起來像是發(fā)生了兩次),一個是數(shù)字索引,另一個是字符串索引。
使用緩存:
要用緩存,你需要用函數(shù)unserialize()來使數(shù)據(jù)還原成原始格式與類型。
你可以用file_get_contents()這個函數(shù)來讀取sql_cache.txt文件的內(nèi)容,把它賦給一個變量。
請注意:這個函數(shù)在PHP4.3.0及以上版本有效。若你使用的是一個老版本的PHP,一個簡單的方法是用file()函數(shù)(讀整個文件到一個數(shù)組,每行變成一個數(shù)組)。implode()函數(shù)用于把數(shù)組的各元素連接成一個字符串然后使用unserialize()反序列化。*/
// file_get_contents() 適合于for PHP < 4.3.0
$file = ‘sql_cache.txt’;
$records = unserialize(implode(”,file($file)));
//現(xiàn)在你可以通過$records數(shù)組并且取得原始查詢的數(shù)據(jù):
foreach ($records as $id=>$row) {
print $row['category_name'].”<br>”;
}
/* 注意$records是數(shù)組(一個包含了查詢結(jié)果的數(shù)字索引列——每行是一個數(shù)字和一個字符串…真是混亂)的一排。
把它們放在一塊:
基于本例子中的時間來決定是否緩存。如果文件修改的時間戳比當(dāng)前時間戳減去過期時間戳大,那么就用緩存,否則更新緩存。
l 檢查文件是否存在并且時間戳小于設(shè)置的過期時間
l 獲取存儲在緩存文件中的記錄或者更新緩存文件
$file = ‘sql_cache.txt’;*/
$expire = 86400; // 24 小時 (單位:秒)
if (file_exists($file)&&filemtime($file) > (time() – $expire)){
// 取得緩存中的記錄
$records = unserialize(file_get_contents($file));
} else {
// 通過 serialize() 函數(shù)創(chuàng)建緩存
}
/*附加其它可能的:
l 把緩存結(jié)果存儲在共享內(nèi)存中以獲取更快的速度
l 增加一個功能隨機(jī)地運行SQL查詢并且檢查是否輸出與緩存輸出一致。如果不一致,則更新緩存(本函數(shù)運行次數(shù)的概率可以定為1/100)。通過哈希算法(如MD5())可以協(xié)助判斷字符串或者文件是否改變。
l 增加一個管理員的功能,人工的刪除這個緩存文件,以強(qiáng)制更新緩存(如file_exists()函數(shù)返回 false時)。你可以用函數(shù)unlink()刪除文件。
腳本:*/
$file = ‘sql_cache.txt’;
$expire = 86400; // 24 小時
if (file_exists($file)&&filemtime($file) > (time() – $expire)) {
$records = unserialize(file_get_contents($file));
//$records=unserialize(fread($file,filesize($file)));
} else {
$link = mysql_connect(‘localhost’,'username’,'password’) or die (mysql_error());
mysql_select_db(‘shop’) or die (mysql_error());
/* 構(gòu)造SQL查詢 */
$query = “SELECT * FROM categories”;
$result = mysql_query($query) or die (mysql_error());
while ($record = mysql_fetch_array($result) ) {
$records[] = $record;
}
$OUTPUT = serialize($records);
$fp = fopen($file,”w”);
//fputs($fp, $OUTPUT);
fwrite($fp,$OUTPUT);
fclose($fp);
} // end else
// 查詢結(jié)果在數(shù)組 $records 中
foreach ($records as $id=>$row) {
if ($row['category_id'] == $_REQUEST['category_id']) {
// 被選擇的目錄顯示粗體字
print ‘<B>’.$row['category_name'].’</B><BR>’;
} else {
// 其它目錄顯示用常規(guī)字體
print $row['category_name'].’<br>’;
}
} // end foreach
實例:
$s = microtime(true);$file = './sql_cache.txt';$class_rs = array();if(file_exists($file)){ $class_rs = unserialize(implode('',file($file)));}if(empty($class_rs)){ $My_rs = mysql_query('select * from tbl_province'); while( $row = mysql_fetch_array($My_rs) ) { $class_rs[$row['ID']] = $row ; } $output = serialize($class_rs); $fp = fopen($file, 'w'); fputs($fp, $output); fclose($fp);}function findChild($arr, $pid){ if(!is_array($arr) || $arr == ""){ return $arr; } $temp = array(); foreach ($arr as $k=>$v){ if($arr[$k]['PID'] == $pid){ $temp[] = $arr[$k]; } } return $temp;}$str = findChild($class_rs, 15);print_r($str);$e = microtime(true);$diff = $e - $s;echo number_format($diff, 10, '.', '')." seconds";PHP編程
鄭重聲明:本文版權(quán)歸原作者所有,轉(zhuǎn)載文章僅為傳播更多信息之目的,如作者信息標(biāo)記有誤,請第一時間聯(lián)系我們修改或刪除,多謝。
新聞熱點
疑難解答