為什么要靜態(tài)化頁面?
當(dāng)用戶訪問一個不經(jīng)常更新的Web頁面,php接到指示對php腳本文件進行解析,從數(shù)據(jù)庫查詢到該頁面所需要的數(shù)據(jù),然后對頁面模板進行渲染,最后將一個成品頁面展示給用戶。
單次請求對于服務(wù)器來說非常簡單,處理起來非??欤侨绻瑫r有成千上萬各用戶請求該頁面呢?這無疑是對資源的一種浪費,這就是我們要做靜態(tài)的目的。
靜態(tài)化分為純靜態(tài)與偽靜態(tài),純靜態(tài)又分為局部純靜態(tài)和全部純靜態(tài)。
偽靜態(tài)
偽靜態(tài)顧名思義,它并不是真的靜態(tài)頁面而是偽裝的。例如一個以php作為后端語言的web站點,正常情況下他的url應(yīng)當(dāng)是類似于http://www.example.com/index.php。
當(dāng)我們做了偽靜態(tài)處理后,當(dāng)你訪問同一個頁面它展示的url可能就是http://www.example.com/index.html了。
它的作用是路由簡化,能夠更好的被搜索引擎收錄,當(dāng)你不想讓用戶知道你的后端語言時也可以采用這種方法。
純靜態(tài)
局部純靜態(tài)
一個頁面通常由多個部分組成,例如一個博客,他可能由正文、分類、友情鏈接、欄目等部分組成。當(dāng)有些部分更新頻繁,而有些部分不常更新時就可以采用局部靜態(tài)化。
全部純靜態(tài)
看了前面的內(nèi)容這個就很容易理解了,當(dāng)一個頁面所有內(nèi)容都不常更新即采用這種方式。
靜態(tài)化頁面實現(xiàn)原理
首先要說的是一個叫做緩沖器(buffer)的東西。
舉個簡單的例子說明它的作用:我們在編輯一篇文檔時,在我們沒有保存之前,系統(tǒng)是不會向磁盤寫入的,而是寫到buffer中,當(dāng)buffer寫滿或者執(zhí)行了保存操作,才會將數(shù)據(jù)寫入磁盤。
對于PHP來說,每一次像 echo 這樣的輸出操作,同樣是先寫入到了 php buffer 里,在腳本執(zhí)行完畢或者執(zhí)行了強制輸出緩存操作,數(shù)據(jù)才會在瀏覽器上顯示。
對于這個緩沖區(qū),在php輸出內(nèi)容之前,我們?nèi)〕鼍彌_區(qū)的內(nèi)容(這里就是渲染好的模板內(nèi)容了),然后將其寫入一個靜態(tài)文件中并設(shè)置過期時間,當(dāng)下次用戶訪問該頁面的時候,如果該靜態(tài)文件存在并且在有效期內(nèi),我們就直接將該靜態(tài)文件展示給用戶看,否則重寫靜態(tài)文件。
代碼實現(xiàn)
數(shù)據(jù)庫連接,用到了單例模式。
Database.php
<?php
class Database {
//用于保存實例化對象
private static $instance;
//用于保存數(shù)據(jù)庫句柄
private $db = null;
//禁止直接實例化,負(fù)責(zé)數(shù)據(jù)庫連接,將數(shù)據(jù)庫連接句柄保存至私有變量$db
private function __construct($options) {
$this->db = MySQLi_connect($options['db_host'], $options['db_user'], $options['db_password'], $options['db_database']);
}
//負(fù)責(zé)實例化數(shù)據(jù)庫類,返回實例化后的對象
public static function getInstance($options) {
if (!(self::$instance instanceof self)) {
self::$instance = new self($options);
}
return self::$instance;
}
//獲取數(shù)據(jù)庫連接句柄
public function db() {
return $this->db;
}
//禁止克隆
private function __clone() {
// TODO: Implement __clone() method.
}
//禁止重構(gòu)
private function __wakeup() {
// TODO: Implement __wakeup() method.
}
}
用于靜態(tài)化頁面
Cache.php
<?php
class Cache {
public function index($options) {
//判斷文件是否存在,判斷是否過期
if (is_file('shtml/index.shtml') && (time() - filemtime('shtml/index.shtml') < 300)) {
require_once ('index.shtml');
}else {
require_once ('Database.php');
$con = Database::getInstance($options)->db();
$sql = "SELECT * FROM pro_test";
$exe_res = mysqli_query($con, $sql);
$res = mysqli_fetch_all($exe_res);
try{
if (!$res) {
throw new Exception("no result");
}
}catch (Exception $e) {
echo 'Message: ' .$e->getMessage();
}
//開啟緩存區(qū),這后面的內(nèi)容都會進緩存區(qū)
ob_start();
//引入模板文件(模板會渲染數(shù)據(jù))
require_once ('templates/index.php');
//取出緩存區(qū)內(nèi)容(在這里是渲染后的模板),將其保存(默認(rèn)會覆蓋原來的)為index.shtml(static html)
file_put_contents('shtml/index.shtml', ob_get_contents());
}
}
}
//數(shù)據(jù)庫配置信息
$options = [
'db_host' => 'mysql',
'db_user' => 'root',
'db_password' => 'localhost',
'db_database' => 'pro_shop',
];
$obj = new Cache();
$obj->index($options);
template/index.php
<!DOCTYPE>
<html>
<head>
<meta charset="UTF-8">
<title>首頁</title>
</head>
<body>
<?php foreach ($res as $item) {?>
<div>姓名:<?php echo $item[1]?></div>
<div>密碼:<?php echo $item[2]?></div>
<?php }?>
</body>
</html>
瀏覽器訪問 localhost/Cache.php


