緩衝SQL查詢結果 為什麼要緩衝查詢結果? 緩衝查詢結果能極大地改進指令碼執行時間和資源需求。 緩衝SQL查詢結果也允許你通過後期處理資料。如果你用檔案快取去儲存全部指令碼的輸出結果(HTML輸出),這樣可能是行不通的。 當你執行一個SQL查詢時,點典的處理過程是: <!--[if !supportLists]-->l <!--[endif]-->串連資料庫 <!--[if !supportLists]-->l <!--[endif]-->準備SQL查詢 <!--[if !supportLists]-->l <!--[endif]-->發送查詢到資料庫 <!--[if !supportLists]-->l <!--[endif]-->取得返回結果 <!--[if !supportLists]-->l <!--[endif]-->關閉資料庫連接 以上方法非常佔用資源並且相反的影響了指令碼的效能。只能通過取得的大量返回資料和資料庫伺服器的位置這二個要素來相互協調。儘管持續串連可以改進串連資料庫時的負載,但非常耗費記憶體資源,如果擷取的是大量的資料,那麼儲存的全部時間會非常短暫。 建立一條SQL查詢: SQL(結構化查詢語言 (SQL))查詢被用作操作資料庫及它內容的介面。SQL可用於定義和編輯表的結構,插入資料到表,更新或刪除表中的資訊。 SQL是用於與資料通訊的語言,在大多數PHP資料庫擴充(MySQL,ODBC,Oracle等)通過傳遞SQL查詢到資料庫中來管理整個過程。 本教程中,僅僅用select語言來擷取資料庫中的資料。這些資料將被緩衝,之後將用作資料來源。 決定什麼時候更新緩衝: 根據程式的需要,緩衝可以採取多種形式。最常見的3種方式是: <!--[if !supportLists]-->l <!--[endif]-->時間觸發緩衝(到期的時間戳記) <!--[if !supportLists]-->l <!--[endif]-->內容改變觸發緩衝(探索資料改變後,相應地更新緩衝) <!--[if !supportLists]-->l <!--[endif]-->人工觸發緩衝(人工的方式告知系統資訊超期並且強制產生新的緩衝) 你的緩衝需求可能是以上原理的一個或多個的綜合。本教程將討論時間觸發方式。然而,在一個全面的緩衝機制中,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 在本例中,通過時間觸發緩衝技術被運用,設定一段時間後讓其緩衝SQL輸出到期。在此特殊的例子中,定一段時間為24小時。 序列化例子: <!--[if !supportLists]-->l <!--[endif]-->串連資料庫 <!--[if !supportLists]-->l <!--[endif]-->執行查詢 <!--[if !supportLists]-->l <!--[endif]-->取得所有結果構成一個數組以便後面你可以訪問 <!--[if !supportLists]-->l <!--[endif]-->序列化數組 <!--[if !supportLists]-->l <!--[endif]-->儲存序列化過的數組到檔案中 [php] view plaincopy $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()還原序列化。 // file_get_contents() 適合於for PHP < 4.3.0 [php] view plaincopy $file = 'sql_cache.txt'; $records = unserialize(implode('',file($file))); //現在你可以通過$records數組並且取得原始查詢的資料: foreach ($records as $id=>$row) { print $row['category_name']."<br>"; }注意$records是數組(一個包含了查詢結果的數字索引列——每行是一個數字和一個字串...真是混亂)的一排。 把它們放在一塊: 基於本例子中的時間來決定是否緩衝。如果檔案修改的時間戳記比目前時間戳減去到期時間戳記大,那麼就用緩衝,否則更新緩衝。 <!--[if !supportLists]-->l <!--[endif]-->檢查檔案是否存在並且時間戳記小於設定的到期時間 <!--[if !supportLists]-->l <!--[endif]-->擷取儲存在快取檔案中的記錄或者更新快取檔案 [php] view plaincopy $file = 'sql_cache.txt'; $expire = 86400; // 24 小時 (單位:秒)if (file_exists($file) && filemtime($file) > (time() - $expire)) { // 取得緩衝中的記錄 $records = unserialize(file_get_contents($file)); } else { // 通過 serialize() 函數建立緩衝 }附加其它可能的: <!--[if !supportLists]-->l <!--[endif]-->把緩衝結果儲存在共用記憶體中以擷取更快的速度 <!--[if !supportLists]-->l <!--[endif]-->增加一個功能隨機地運行SQL查詢並且檢查是否輸出與緩衝輸出一致。如果不一致,則更新緩衝(本函數運行次數的機率可以定為1/100)。通過雜湊演算法(如MD5())可以協助判斷字串或者檔案是否改變。 <!--[if !supportLists]-->l <!--[endif]-->增加一個管理員的功能,人工的刪除這個快取檔案,以強制更新緩衝(如file_exists()函數返回false時)。你可以用函數unlink()刪除檔案。 指令碼: [php] view plaincopy $file = 'sql_cache.txt'; $expire = 86400; // 24 小時if (file_exists($file) && filemtime($file) > (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=>$row) { if ($row['category_id'] == $_REQUEST['category_id']) { // 被選擇的目錄顯示粗體字 print '<B>'.$row['category_name'].'</B><BR>'; } else { // 其它目錄顯示用常規字型 print $row['category_name'].'<br>'; } } // end foreach
|