PHP中的SQL查詢緩存

適合讀者php

本教程適合於那些對緩存SQL查詢以減小數據庫鏈接與執行的負載、提升腳本性能感興趣的PHP程序員。html

概述mysql

許多站點使用數據庫做爲站點數據存儲的容器。數據庫包含了產器信息、目錄結構、文章或者留言本,有些數據極可能是徹底靜態的,這些將會從一個緩存系 統中獲得的極大好處。 這樣一個系統經過把SQL查詢的結果緩存到系統的一個文件中存儲,從而阻止鏈接數據庫,構造查詢與取得返回結果而提升了響應時間。 有些系統數據庫並非放在WEB服務器上的,這樣須要一個遠程鏈接(TCP或者其它相似的),或者從數據庫中獲取大量的數據,這樣你得忍受更多時間,這決 定於系統響應時間與資源利用。程序員

前提算法

本教程使用MySQL做爲數據庫。你須要安裝MySQL(www.mysql.com下載是有效的)和激活PHP MYSQL擴展(默認狀況是激活的)。 因爲要查詢數據庫,你須要知識一些SQL(結構化查詢語言)的基本常識。sql

緩存SQL查詢結果數據庫

爲何要緩存查詢結果? 緩存查詢結果能極大地改進腳本執行時間和資源需求。 緩存SQL查詢結果也容許你經過後期處理數據。若是你用文件緩存去存儲所有腳本的輸出結果(HTML輸出),這樣多是行不通的。 當你執行一個SQL查詢時,點典的處理過程是:數組

l        鏈接數據庫緩存

l        準備SQL查詢服務器

l        發送查詢到數據庫

l        取得返回結果

l        關閉數據庫鏈接

以上方法很是佔用資源而且相反的影響了腳本的性能。只能經過取得的大量返回數據和數據庫服務器的位置這二個要素來相互協調。儘管持續鏈接能夠改進鏈接數據庫時的負載,但很是耗費內存資源,若是獲取的是大量的數據,那麼存儲的所有時間會很是短暫。

建立一條SQL查詢:

SQL(結構化查詢語言)查詢被用做操做數據庫及它內容的接口。SQL可用於定義和編輯表的結構,插入數據到表,更新或刪除表中的信息。 SQL是用於與數據通信的語言,在大多數PHP數據庫擴展(MySQL,ODBC,Oracle等)經過傳遞SQL查詢到數據庫中來管理整個過程。 本教程中,僅僅用select語言來獲取數據庫中的數據。這些數據將被緩存,以後將用做數據源。

決定何時更新緩存:

根據程序的須要,緩存能夠採起多種形式。最多見的3種方式是:

l        時間觸發緩存(過時的時間戳)

l        內容改變觸發緩存(發現數據改變後,相應地更新緩存)

l        人工觸發緩存(人工的方式告知系統信息超期而且強制產生新的緩存)

你的緩存需求多是以上原理的一個或多個的綜合。本教程將討論時間觸發方式。然而,在一個全面的緩存機制中,3種方式的綜合將被使用。

緩存結果:

基本的緩存是用PHP的兩個函數serialize()和unserialize()(譯註:這二個函數分別表明序列化與反序列化)。 函數serialize()用於存儲PHP的值,它能保證不失去這些值的類型和結構。 事實上,PHP的session擴展是用序列化過的變量,把session變量($_SESSION)存儲在系統的一個文件中。 函數unserialize()與以上操做相反而且使序列化過的字符串返回到它原來的結構和數據內容。 在本例中,以一個電子商務商店爲例。商店有2個基本表,categories和products(此處爲原始數據庫表名).product表可能天天都在 變化,categories仍然是不變靜止的。 要顯示產品,你能夠用一個輸出緩存腳原本存儲輸出的HTML結果到一個文件中。然而categories表可能須要後期處理。例如,全部的目錄經過變量 category_id(經過$_REQUEST['category_id']來取得)被顯示,你可能但願高亮當前被選擇的目錄。

表categories結構

Field Type Key Extra
category_id category_name category_description int(10) unsigned varchar(255) text PRI auto_incremen

在本例中,經過時間觸發緩存技術被運用,設定一段時間後讓其緩存SQL輸出過時。在此特殊的例子中,定一段時間爲24小時。 序列化例子: l        鏈接數據庫 l        執行查詢 l        取得全部結果構成一個數組以便後面你能夠訪問 l        序列化數組 l        保存序列化過的數組到文件中

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
$file = 'sql_cache.txt';
 
$link = mysql_connect('localhost','username','password')
 
or die (mysql_error());
 
mysql_select_db('shop')
 
or die (mysql_error());
 
/* 構造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);
 
fclose($fp);

查看sql_cache.txt文件,裏面的內容可能相似這樣的: 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」;}} 這個輸出是它的變量和類型的內部表現形式。倘若你用mysql_fetch_array()函數返回數字索引的數組和一個關聯的數組(這就是爲何數據看 起來像是發生了兩次),一個是數字索引,另外一個是字符串索引。

使用緩存:

要用緩存,你須要用函數unserialize()來使數據還原成原始格式與類型。 你能夠用file_get_contents()這個函數來讀取sql_cache.txt文件的內容,把它賦給一個變量。 請注意:這個函數在PHP4.3.0及以上版本有效。若你使用的是一個老版本的PHP,一個簡單的方法是用file()函數(讀整個文件到一個數組,每行 變成一個數組)。implode()函數用於把數組的各元素鏈接成一個字符串而後使用unserialize()反序列化。

1
2
3
4
5
6
7
8
9
10
11
12
13
// file_get_contents() 適合於for PHP < 4.3.0
 
$file = 'sql_cache.txt';
 
$records = unserialize(implode('',file($file)));
 
如今你能夠經過$records數組而且取得原始查詢的數據:
 
foreach ($records as $id=&gt;$row) {
 
print $row['category_name']."<br>";
 
}

注意$records是數組(一個包含了查詢結果的數字索引列——每行是一個數字和一個字符串…真是混亂)的一排。

把它們放在一塊:

基於本例子中的時間來決定是否緩存。若是文件修改的時間戳比當前時間戳減去過時時間戳大,那麼就用緩存,不然更新緩存。 l        檢查文件是否存在而且時間戳小於設置的過時時間 l        獲取存儲在緩存文件中的記錄或者更新緩存文件 $file = ’sql_cache.txt’; $expire = 86400; // 24 小時 (單位:秒) if (file_exists($file) && filemtime($file) > (time() – $expire)) { // 取得緩存中的記錄 $records = unserialize(file_get_contents($file)); } else { // 經過 serialize() 函數建立緩存 }

附加其它可能的:

l        把緩存結果存儲在共享內存中以獲取更快的速度 l        增長一個功能隨機地運行SQL查詢而且檢查是否輸出與緩存輸出一致。若是不一致,則更新緩存(本函數運行次數的機率能夠定爲1/100)。經過哈希算法 (如MD5())能夠協助判斷字符串或者文件是否改變。 l        增長一個管理員的功能,人工的刪除這個緩存文件,以強制更新緩存(如file_exists()函數返回false時)。你能夠用函數unlink()刪 除文件。

腳本:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
$file = 'sql_cache.txt';
$expire = 86400; // 24 小時
if (file_exists($file) &amp;&amp;filemtime($file) &gt; (time() - $expire)) {
 
$records = unserialize(file_get_contents($file));
 
} else {
 
$link = mysql_connect('localhost','username','password') or die (mysql_error());
 
mysql_select_db('shop') or die (mysql_error());
 
/* 構造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);
 
fclose($fp);
 
}
 
// end else
 
// 查詢結果在數組 $records 中
 
foreach ($records as $id=&gt;$row) {
 
if ($row['category_id'] == $_REQUEST['category_id']) {
 
// 被選擇的目錄顯示粗體字
 
print '<B>'.$row['category_name'].'</B><BR>';
 
} else {
 
// 其它目錄顯示用常規字體 print $row['category_name'].'<br>';
 
} } // end foreach
閱讀全文
類別: php專欄  查看評論

相關文章
相關標籤/搜索