PHP的自定義模板引擎
前面的話
在大多數的項目組中,開發一個Web程序都會出現這樣的流程:計劃文檔提交之後,前端工程師制作了網站的外觀模型,然後把它交給後端工程師,它們使用後端代碼實現程序邏輯,同時使用外觀模型做成基本架構,然後工程被返回到前端工程師繼續完善。就這樣工程可能在後端工程師和前端工程師之間來來回回好幾次。由於後端工程師不幹預任何相關HTML標簽,同時也不需要前端代碼和後端代碼混合在一起。前端工程師只需要配置文件,動態區塊和其他的界面部分,不必要去接觸那些錯綜復雜的後端代碼。因此,這時候有一個很好的模板支持就顯得很重要了。本文將詳細介紹PHP中的模板引擎
概述
什麽是網站模板?準確地說,是指網站頁面模板,即每個頁面僅是一個板式,包括結構、樣式和頁面布局,是創建網頁內容的樣板,也可以理解為已有的網頁框架。可以將模板中原有的內容替換成從服務器端數據庫中動態內容,目的是可以保持頁面風格一致
PHP是一種HTML內嵌式的在服務器端執行的腳本語言,所以大部分PHP開發出來的Web應用,初始的開發模板就是混合層的數據編程。雖然通過MVC設計模式可以把程序應用邏輯與網頁呈現邏輯強制性分離,但也只是將應用程序的輸入、處理和輸出分開,網頁呈現邏輯(視圖)還會有HTML代碼和PHP程序強耦合在一起。PHP腳本的編寫者必須既是網頁設計者,又是PHP開發者
現在已經有很多解決方案,可以將網站的頁面設計和PHP應用程序幾乎完全分離。這些解決方案稱為“模板引擎”,它們正在逐步消除由於缺乏層次分離而帶來的難題。模板引擎的目的,就是要達到上述提到的邏輯分離的功能。它能讓程序開發者專註於資料的控制或是功能的達成。因此,模板引擎很適合公司的Web開發團隊使用,使每個人都能發揮其專長
模板引擎技術的核心比較簡單。只要將前端頁面指定為模板文件,並將這個模板文件中動態的內容,如數據庫輸出、用戶交互等部分,定義成使用特殊“定界符”包含的“變量”,然後放在模板文件中相應的位置。當用戶瀏覽時,由PHP腳本程序打開該模板文件,並將模板文件中定義的變量進行替換。這樣,模板中的特殊變量被替換為不同的動態內容時,就會輸出需要的頁面
目前,可以在PHP中應用的並且比較成熟的模板有很多,例如Smarty、PHPLIB、IPB等幾十種。使用這些通過PHP編寫的模板引擎,可以讓代碼脈絡更加清晰,結構更加合理化。也可以讓網站的維護和更新變得更容易,創造一個更加良好的開發環境,讓開發和設計工作更容易結合在一起。但是,沒有哪一個PHP模板是最合適、最完美的。因為PHP模板就是大眾化的東西,並不是針對某個人開發的。如果能在對模板的特點、應用有清楚的認識基礎上,充分認識到模板的優勢劣勢,就可以知道是否選擇使用模板引擎或選擇使用哪個模板引擎
自定義模板引擎類
自定義模板引擎,能夠更好的掌握模板引擎的工作機制,為學習Smarty做好準備。更重要的是,屬於自己的PHP模板引擎永遠不是固定不變的,可以根據項目的需要為其量身定制
在下例中,通過前面介紹的模板引擎概念創建了屬於自己的一個簡單模板引擎,可以用來處理模板的基本功能。例如:變量替換、分支結構、數組循環遍歷,以及模板之間相互嵌套等,如下所示:
<?php /** file: mytpl.class.php 類名為MyTpl是自定義的模板引擎 通過該類對象加載模板文件並解析,將解析後的結果輸出 */ class Mytpl { public $template_dir = ‘templates‘; //定義模板文件存放的目錄 public $compile_dir = ‘templates_c‘; //定義通過模板引擎組合後文件存放目錄 public $left_delimiter = ‘<{‘; //在模板中嵌入動態數據變量的左定界符號 public $right_delimiter = ‘}>‘; //在模板中嵌入動態數據變量的右定界符號 private $tpl_vars = array(); //內部使用的臨時變量 /** 將PHP中分配的值會保存到成員屬性$tpl_vars中,用於將模板中對應的變量進行替換 @param string $tpl_var 需要一個字符串參數作為關聯數組下標,要和模板中的變量名對應 @param mixed $value 需要一個標量類型的值,用來分配給模板中變量的值 */ function assign($tpl_var, $value = null) { if ($tpl_var != ‘‘) $this->tpl_vars[$tpl_var] = $value; } /** 加載指定目錄下的模板文件,並將替換後的內容生成組合文件存放到另一個指定目錄下 @param string $fileName 提供模板文件的文件名 */ function display($fileName) { /* 到指定的目錄中尋找模板文件 */ $tplFile = $this->template_dir.‘/‘.$fileName; /* 如果需要處理的模板文件不存在,則退出並報告錯誤 */ if(!file_exists($tplFile)) { die("模板文件{$tplFile}不存在!"); } /* 獲取組合的模板文件,該文件中的內容都是被替換過的 */ $comFileName = $this->compile_dir."/com_".$fileName.‘.php‘; /* 判斷替換後的文件是否存在或是存在但有改動,都需要重新創建 */ if(!file_exists($comFileName) || filemtime($comFileName) < filemtime($tplFile)) { /* 調用內部替換模板方法 */ $repContent = $this->tpl_replace(file_get_contents($tplFile)); /* 保存由系統組合後的腳本文件 */ file_put_contents($comFileName, $repContent); } /* 包含處理後的模板文件輸出給客戶端 */ include($comFileName); } /** 內部使用的私有方法,使用正則表達式將模板文件‘<{ }>‘中的語句替換為對應的值或PHP代碼 @param string $content 提供從模板文件中讀入的全部內容字符串 @return $repContent 返回替換後的字符串 */ private function tpl_replace($content) { /* 將左右定界符號中,有影響正則的特殊符號轉義 例如,<{ }>轉義\<\{ \}\> */ $left = preg_quote($this->left_delimiter, ‘/‘); $right = preg_quote($this->right_delimiter, ‘/‘); /* 匹配模板中各種標識符的正則表達式的模式數組 */ $pattern = array( /* 匹配模板中變量 ,例如,"<{ $var }>" */ ‘/‘.$left.‘\s*\$([a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)\s*‘.$right.‘/i‘, /* 匹配模板中if標識符,例如 "<{ if $col == "sex" }> <{ /if }>" */ ‘/‘.$left.‘\s*if\s*(.+?)\s*‘.$right.‘(.+?)‘.$left.‘\s*\/if\s*‘.$right.‘/ies‘, /* 匹配elseif標識符, 例如 "<{ elseif $col == "sex" }>" */ ‘/‘.$left.‘\s*else\s*if\s*(.+?)\s*‘.$right.‘/ies‘, /* 匹配else標識符, 例如 "<{ else }>" */ ‘/‘.$left.‘\s*else\s*‘.$right.‘/is‘, /* 用來匹配模板中的loop標識符,用來遍歷數組中的值, 例如 "<{ loop $arrs $value }> <{ /loop}>" */ ‘/‘.$left.‘\s*loop\s+\$(\S+)\s+\$([a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)\s*‘.$right.‘(.+?)‘.$left.‘\s*\/loop\s*‘.$right.‘/is‘, /* 用來遍歷數組中的鍵和值,例如 "<{ loop $arrs $key => $value }> <{ /loop}>" */ ‘/‘.$left.‘\s*loop\s+\$(\S+)\s+\$([a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)\s*=>\s*\$(\S+)\s*‘.$right.‘(.+?)‘.$left.‘\s*\/loop \s*‘.$right.‘/is‘, /* 匹配include標識符, 例如,‘<{ include "header.html" }>‘ */ ‘/‘.$left.‘\s*include\s+[\"\‘]?(.+?)[\"\‘]?\s*‘.$right.‘/ie‘ ); /* 替換從模板中使用正則表達式匹配到的字符串數組 */ $replacement = array( /* 替換模板中的變量 <?php echo $this->tpl_vars["var"]; */ ‘<?php echo $this->tpl_vars["${1}"]; ?>‘, /* 替換模板中的if字符串 <?php if($col == "sex") { ?> <?php } ?> */ ‘$this->stripvtags(\‘<?php if(${1}) { ?>\‘,\‘${2}<?php } ?>\‘)‘, /* 替換elseif的字符串 <?php } elseif($col == "sex") { ?> */ ‘$this->stripvtags(\‘<?php } elseif(${1}) { ?>\‘,"")‘, /* 替換else的字符串 <?php } else { ?> */ ‘<?php } else { ?>‘, /* 以下兩條用來替換模板中的loop標識符為foreach格式 */ ‘<?php foreach($this->tpl_vars["${1}"] as $this->tpl_vars["${2}"]) { ?>${3}<?php } ?>‘, ‘<?php foreach($this->tpl_vars["${1}"] as $this->tpl_vars["${2}"] => $this->tpl_vars["${3}"]) { ?>${4}<?php } ?>‘, /*替換include的字符串*/ ‘file_get_contents($this->template_dir."/${1}")‘ ); /* 使用正則替換函數處理 */ $repContent = preg_replace($pattern, $replacement, $content); /* 如果還有要替換的標識,遞歸調用自己再次替換 */ if(preg_match(‘/‘.$left.‘([^(‘.$right.‘)]{1,})‘.$right.‘/‘, $repContent)) { $repContent = $this->tpl_replace($repContent); } /* 返回替換後的字符串 */ return $repContent; } /** 內部使用的私有方法,用來將條件語句中使用的變量替換為對應的值 @param string $expr 提供模板中條件語句的開始標記 @param string $statement 提供模板中條件語句的結束標記 @return strin 將處理後的條件語句相連後返回 */ private function stripvtags($expr, $statement=‘‘) { /* 匹配變量的正則 */ $var_pattern = ‘/\s*\$([a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*)\s*/is‘; /* 將變量替換為值 */ $expr = preg_replace($var_pattern, ‘$this->tpl_vars["${1}"]‘, $expr); /* 將開始標記中的引號轉義替換 */ $expr = str_replace("\\\"", "\"", $expr); /* 替換語句體和結束標記中的引號 */ $statement = str_replace("\\\"", "\"", $statement); /* 將處理後的條件語句相連後返回 */ return $expr.$statement; } } ?>
在Mytpl類中聲明的多個方法中,除被封裝過的方法之外,只有兩個公有方法assign()和display()在創建對象以後可以被雕用。其中assign()方法用來將PHP腳本中的數據分配給模板中對應的變量,display()方法則用來將特定的templates目錄下的模板文件加載到PHP腳本中。同時將模板文件中使用“<{”和“>>”標記聲明的自定義模板語句,匹配出來並替換成相對應的PHP語法格式,然後將替換後的內容保存在特定的templates_c目錄下。在運行時還要編譯成一個非模板技術的PHP文件,並將其以模板文件名加上“com_”前綴和“.php”的擴展名形式保存。再通過include()函數將處理後的模板文件包含,並使用PHP解析後發送給客戶端
使用模板引擎
使用自定義的模板引擎比較容易,都是自己定義的語法格式。但要記住,所有流行的模板引繁解決方案都遵循同樣的一組相同的核心實現原則,就是與編程語言一樣,學習了一種語言就可以更容易地掌握其他語言。使用模板引擎最主要的原因就是將前端工程師和後端工程師的工作分開,所以模板引擎不僅後端工程師需要使用,前端工程師也需要使用
1、後端工程師對模板引擎的使用
在PHP腳本中包含模板引擎類所在的文件。如下所示:
require("mytpl.class.php"); //包含模板引擎類,相當於模板引擎安裝
創建模板引擎類的對象並對一些成員屬性進行初始化賦值。如下所示:
$tpl=new MyTpl(); //創建模板引擎類的對象,也可以根據參數對成員初始化
將動態數據(包括標量和數組類型的數據,例如從數據庫的表中獲得的數據數組)使用模板引擎對象中的assign()方法分配給模板文件,這個方法可以使用多次,將任意多個變量分配給模板。如下所示:
$tpl->assign("var","this is a value"); //可以分配標量類型數據,可以使用多次 $tpl->assign("arr",array(array(1,2),array("a","b"))); //也可以分配數組包括多維數組
在PHP腳本中通過調用模板對象中的display()方法,並將模板文件名作為參數傳入,就會加載指定目錄中對應的模板文件到PHP腳本中。再通過模板引擎中的替換方法對模板中自定義的語法進行解析,然後輸出處理後的模板。如下所示:
$tpl->display("test.tpl"); //參數“test.tpl”為特定目錄下的模板文件
2、前端工程師對模板引擎的使用
前端工程師需要將編寫的模板文件存放到指定的目錄中,這個目錄是通過在模板對象中使用$template_dir屬性指定的,默認的設置是當前目錄下的“templates”目錄。另外,模板文件的命名以及後綴名的設置可以隨意,例如index.tpl、test.htm、header.tp;等
模板文件是通過使用HTML、CSS以及javascript等Web前臺語言以編寫的純靜態負而。但可以在模板文件中使用“<{”和“}>”兩個分隔符中間定義一個變量(類似PHP中的變量格式),該變量可以接受並輸出由PHP腳本中分配過來的動態數據。在模板中使用的“<{”和“}>”兩個分隔符對,也可以根據個人愛好在模板引擎類中修改。如下所示:
姓名:<{$name}>,年齡:<{$age}>,性別:<{$sex}> //模板中使用占位符
如果在PHP腳本中是將數組分配給模板,也可以在模板中進行遍歷,還可以通過嵌套的方式遍歷多維數組。使用的是在模板引擎中定義的“<{loop}>”標記對,使用的方式和PHP中foreach結構的語法格式相似。如下所示:
<{loop $arr $value }> //遍歷數組$arr中的元素值 數組中的元素值<{$value}> //每次遍歷輸出元素中的值 <{/loop}> //在模板中遍歷數組的結束標記 <{loop $arr $key=>$value }> //遍歷數組$arr中的元素下標和元素值 數組中的元素鍵<{$key}> //每次遍歷輸出元素中的下標 數組中的元素值<{$value}> //每次遍歷輸出元素中的值 <{/loop}> //在模板中遍歷數組的結束標記 <{loop $arr $value }> //遍歷數組$arr中的元素值 <{loop $arr $data }> //使用嵌套標記遍歷二維數組 數組中的元素值<{$value}> //每次遍歷輸出元素中的值 <{/loop}> //在模板中遍歷數組的內層結束標記 <{/loop}> //在模板中遍歷數組的外層結束標記
模板引擎還可以解析在模板文件中使用特殊標記編寫的分支結構,語法風格也是和PHP的分支結構類似。是通過在模板文件中使用“<{if}>”標記對實現選擇結構,也可以實現多路分支和嵌套分支的選擇結構。如下所示:
<{if($var=="red")}> <p style="color:red">這是“紅色”的字</p> <{elseif($var=="green")}> <p style="color:green">這是“綠色”的字</p> <{else}> <{if($size=20)}> <p style="font-size:20">這是“20px”大小的字</p> <{/if}> <{/if}>
在自定義的模板引擎中,也添加了在模板文件中包含其他模板文件的功能。可以使用“<{include‘子模板文件名‘}>”標記將子模板包含到當前模板中,還支持在子模板中再次包括另外的子模板。如下所示:
<{include ‘other.tpl‘ }>
使用示例分析
通過在程序中加載模板引擎可以將前端語言與後端語言的代碼分開。首先在PHP程序中獲取數據庫中存儲的數據,再通過加載模板引擎將數據分配出去,然後將模板文件再通過模板引擎加載並處理後輸出。所以PHP程序只是創建動態數據,加載模板引擎並將動態數據分配給模板,完成了PHP程序的工作。而模板的設汁也只需要前端工程師獨立完成,使用HTML、CSS及javascript等前臺頁面設計語言編寫。另外,在模板文件中還需要使用模板引擎可以解析的標記,將PHP中分配過來的動態數據在模板中引用
1、數據庫的設計
假設數據庫服務器在“localhost”主機上,連接的用戶名和密碼分別為“admin”和“123456”,在該服務器上創建一個名為“mydb”的數據庫,並在該數據庫中創建一個名為“User”的用戶表。創建該表的SQL査詢語句如下所示:
CREATE TABLE User( id SMALLINT(3) NOT NULL AUTO_INCREMENT, name VARCHAR(10) NOT NULL DEFAULT ‘‘, sex VARCHAR(4) NOT NULL DEFAULT ‘‘, age SMALLINT(2) NOT NULL DEFAULT ‘0‘, email VARCHAR(20) NOT NULL DEFAULT ‘‘, PRIMARY KEY (id) );
用戶表User創建完成以後,接著可以向該表中插入一些數據作為示例演示使用,SQL查詢語句如下所示:
INSERT INTO User (name,sex,age,email) VALUES ("a","男",27,"[email protected]"), ("b","女",22,"[email protected]"), ("c","女",30,"[email protected]"), ("d","女",24,<a href="mailto:[email protected]" rel="external nofollow">[email protected]</a>);
2、模板的設計
模板的設計不要出現任何的PHP代碼,可以由前端人員來完成。在自定義的模板引擎中,規定了要到指定的目錄中去尋找模板文件,這個特定的目錄可以在創建模板引擎對象時指定,也可以使用默認的目錄設置,默認可以將模板文件存放在當前目錄中的“templates”目錄下。本例共需要三個模板文件main.tpl、header.tpl和footer.tpl,都存放在這個默認的目錄設置中。這三個模板文件的代碼如下所示:
模板的頭部文件header.tpl
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title> <{$title}> </title> </head> <body>
模板的尾部文件footer.tpl
<div style="width:200px;margin: 0 auto;">##### <{$author}> #####</div> </body> </html>
主模板文件main.tpl
<{include ‘header.tpl‘}> <table border="1" align="center" width="500"> <{ loop $users $user }> <tr> <{loop $user $u }> <{if $u == "男" }> <td style="color:green"> <{elseif $u == "女"}> <td style="color:red"> <{else}> <td> <{/if}> <{$u}></td> <{/loop}> </tr> <{/loop}> </table> <{include ‘footer.tpl‘}>
文件main.tpl是主模板文件,在該文件中使用<{include"header.tpl"}>和<{include"footer.tpl"}>兩個標記分別在該文件的頂部和底部,將獨立的頭部和尾部模板文件包含到這個主模板文件中。並在該文件中使用<{tableName}>標記獲取從PHP中動態分配過來的表名,以及使用雙層<{loop}>標記嵌套,遍歷從PHP中動態分配過來的在數據庫中獲取到的二維數組$Users,還在<{loop}>標記中使用條件選擇標記<{if}>組合,將數據中性別為“男”的表格背景設置為紅色和一些其他判斷。被包含進來的頭部模板文件header.tpl和尾部模板文件footer.tpl也同樣可以獲取從PHP中動態分配給模板的數據
3、PHP程序設計
通過模板引擎的使用,PHP程序員在編寫代碼時,只需要PHP一種語言就可以了,不用再去使用HTML、CSS以及javascript等頁面設計語言完成前端的工作了。下面是一個文件名為index.php的PHP腳本文件,和模板引擎類所在的文件mytpl_class.php在同一個目錄下。代碼如下所示:
<?php //包含模板引擎類 include "mytpl.class.php"; //創建模板引擎對象 $tpl = new Mytpl; //連接數據庫 $pdo = new PDO("mysql:host=localhost;dbname=mydb", "admin", "123456"); //執行SQL語句 $stmt = $pdo -> prepare("select id, name, sex,age,email from User order by id"); $stmt ->execute(); $data = $stmt -> fetchAll(PDO::FETCH_ASSOC); //這是從數據庫獲取的動態數據,需要在模板中顯示 $tpl->assign(‘title‘,"自定義模板引擎");$tpl->assign(‘auto‘,"小火柴"); $tpl->assign(‘users‘,$data); $tpl -> display("main.tpl"); ?>
在上面的PHP腳本文件中,通過PDO對象連接MySQL服務器,並獲取用戶表User中的全部記錄,並以PHP的二維數組變量形式保存在變量data中。接著使用包含進來的當前目錄下的“mytplclss.php”文件,創建並初始化模板引擎類的對象data中。接著使用包含進來的當前目錄下的“mytplclss.php”文件,創建並初始化模板引擎類的對象tpl。再通過該對象中的assign()方法向模板分配一些數據,然後使用該對象中的display()方法載入模板文件main.tpl。並將模板中標記的特殊變量替換為從PHP中分配的動態數據,處理完畢以後輸出模板頁面。頁面的輸出結果如下所示
限於各種不同的條件限制,比如時間、經驗,做一個自定義的PHP模板引擎是非常困難的。其實,需要的並不是重新構造一個PHP模板,而是選擇一個最貼近自己的PHP模板加以改造
PHP的自定義模板引擎