首頁 > 後端開發 > php教程 > php資料庫快取實作思路

php資料庫快取實作思路

不言
發布: 2023-03-28 19:08:01
原創
4132 人瀏覽過

這篇文章主要介紹了關於php資料庫快取實現思路,有著一定的參考價值,現在分享給大家,有需要的朋友可以參考一下

快取SQL查詢結果

為什麼要快取查詢結果?

快取查詢結果能大幅改善腳本執行時間和資源需求。

快取SQL查詢結果也允許你透過後製資料。如果你用檔案快取去儲存全部腳本的輸出結果(HTML輸出),這樣可能是行不通的。

當你執行一個SQL查詢時,點典的處理過程是:

l        連接資料庫

l        準備SQL查詢

< !--[if !supportLists]-->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結構

FieldType KeyExtracategory_idcategory_namecategory_descriptionint(10) unsignedvarchar(255)textPRIauto_incremen 

在本例中,透過時間設定一段時間快取技術來設定,SQLSQL快取一個時間後讓其快取完成時間。在此特別的例子中,定一段時間為24小時。

序列化範例:

l        連結資料庫

l        執行查詢

l        取得所有結果構成一個陣列以便後面你可以造訪

l        序列化陣列

保存序列化過的陣列到檔案中

[php] view plaincopy

$file = &#39;sql_cache.txt&#39;;  
$link = mysql_connect(&#39;localhost&#39;,&#39;username&#39;,&#39;password&#39;)  
    or die (mysql_error());  
mysql_select_db(&#39;shop&#39;)  
    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()反序列化。

// file_get_contents() 適合for PHP < 4.3.0

[php] view plaincopy

$file = &#39;sql_cache.txt&#39;;  
$records = unserialize(implode(&#39;&#39;,file($file)));  
//现在你可以通过$records数组并且取得原始查询的数据:  
foreach ($records as $id=>$row) {  
    print $row[&#39;category_name&#39;]."<br>";  
}
登入後複製

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

把它們放在一塊:

基於本範例中的時間來決定是否快取。如果檔案修改的時間戳比目前時間戳記減去過期時間戳大,那麼就用緩存,否則更新快取。

l        檢查檔案是否存在且時間戳小於設定的過期時間

l        取得儲存在快取檔案中的記錄或更新快取檔案

#[

#[ php]

 view plaincopy

$file = &#39;sql_cache.txt&#39;;  
$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()刪除檔案。

#腳本:

[php]

 view plaincopy

$file = &#39;sql_cache.txt&#39;;  
$expire = 86400; // 24 小时
if (file_exists($file) &&  
    filemtime($file) > (time() - $expire)) {  
    $records = unserialize(file_get_contents($file));  
} else {  
    $link = mysql_connect(&#39;localhost&#39;,&#39;username&#39;,&#39;password&#39;)  
        or die (mysql_error());  
    mysql_select_db(&#39;shop&#39;)  
        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=>$row) {  
    if ($row[&#39;category_id&#39;] == $_REQUEST[&#39;category_id&#39;]) {  
        // 被选择的目录显示粗体字  
        print &#39;<B>&#39;.$row[&#39;category_name&#39;].&#39;</B><BR>&#39;;  
    } else {  
        // 其它目录显示用常规字体  
        print $row[&#39;category_name&#39;].&#39;<br>&#39;;  
    }  
} // end foreach
登入後複製
# ######

以上是php資料庫快取實作思路的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
作者最新文章
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板