在PHP世界中選擇最合適的模板

在PHP世界中選擇最合適的模板
比較PHPLIB Template和FastTemplate

王晨 ([email protected])
2001 年 7 月

PHP工程中的模板應用,是進行中型乃至大型項目中建議採用的處理表現層的好辦法。但是具體到模板的實施,採用何種現有的模板技術卻需要進行一番比較。

PHP世界中比較受關注的模板處理有PHPLIB Template和FastTemplate兩種,我們對技術的易用性和速度進行了評測--想知道結果嗎?

事情的起因:你用過FastTemplate嗎?
對於PHP工程中的模板應用,其實我和我的同事們已經在許多的項目中接觸過--關於它的好處,我想無論是在實際開發階段還是上升到設計模式的角度都已經有很多"前輩先哲"討論過了。就項目實施而言,在一些中型甚至大型的項目中,有效的將HTML(還有其他文本形式的表現層)和PHP代碼分開,不僅在開發階段可以分別提高界面設計人員和應用程序編寫人員的工作效率,更會給項目的測試和維護帶來巨大的便利。

但是--本文的目的不是討論模板的優缺點,也不是作爲指導性的教程講授如何在PHP項目中使用模板,而是以應用的視角比較兩種PHP世界中最爲流行的模板處理方式(其實只不過是兩種模板類):PHPLIB Template和FastTemplate。

其實我一直都在"安靜"的使用着PHPLIB Template--很穩定而且看上去速度也不錯,以至於我並不想再去不安的尋找可能更好的替代品--雖然我也知道這個地球上還有FastTemplate這樣的東西(而且還在Perl的世界中大名鼎鼎)。直到有一天,有一個同事問我:"不知道FastTemplate怎麼樣?爲什麼我們不試試FastTemplate呢?"

"好吧,就讓我們試試!"不過作爲一個穩妥的方法,在任何新的模式或者方法引入項目之前,最好能夠更加全面的瞭解它,以及找到一個或者幾個足夠說服自己和同事去採用它的理由--對於FastTemplate也不例外。

主角出場:瞭解PHPLIB Template以及FastTemplate
前面已經說過,我已經使用PHPLIB有一段時間了--我想屏幕前的你也許和我一樣,也對這個優秀的工具類庫印象很深吧!同樣,當我開始尋求模板的解決辦法時,很自然的就會在最接近身邊的工具箱裏搜尋,於是我找到了PHPLIB中的Template類。在最初的很快瀏覽完它提供的API之後(當然還得感謝PHPLIB詳盡的文檔),我就開始了使用它的歷程--直到現在。

而FastTemplate似乎名氣更響亮一些,在其發跡的Perl世界中自然是這樣,在PHP世界中似乎也是,單單從這一點上就足夠讓人相信它的能力了。

關於兩者的使用辦法,本來我想在這裏多廢話幾句的;但是畢竟覺得自己恐怕專門寫出兩篇教程來也沒有現有的教程受歡迎--在本文的參考資料中有關於PHPLIB Template和FastTemplate的有名教程,如果你自認還沒有對這兩種模板或者其中的一種有所認識,建議你先去看看那兩篇文章,應該會得到不少有益的模板應用知識。

(一番鼠標點擊以及眼球轉動甚至親自編寫測試代碼之後,)現在你對兩種模板都有了一些瞭解,也許已經發現了它們之間的很多相似之處,在下面我就會將這些地方歸納一下。

  1. 變量的設置
    很明顯,{FOO}或者{BAR}的形式在兩種模板中都是指定的形式;也就是說,兩種模板處理方式中,模板文件本身的外貌應該可以是一致的(比如都是HTML文件中間含有將要被替換的以{}標識的變量)。
  2. 模板類的初始化(類的構建器)
    都需要在構建模板類的時候指定模板文件存在的目錄位置。
  3. 變量的替換
    模板處理中最常用的就是變量替換,兩種方式除了方法名不同之外(PHPLIB Template採用set_var(),而FastTemplate採用assign()),用法幾乎也是一致的--可以採用(key, value)的方式,也可以直接傳遞一個數組(array(key=>value))。
  4. 模板文件的處理
    都是採用爲每一個模板文件指定一個句柄(handler)的辦法,同時句柄也可以作爲變量的值替換另一模板文件中的變量。
  5. 解析、輸出過程
    都是需要調用parse()方法(這個方法名竟然是相同的)將需要輸出的模板文件解析後賦值給一個句柄,然後調用各自輸出的方法(PHPLIB Template中是p(),FastTemplate中是FastPrint())輸出該句柄的內容並結束處理。
  6. 重複解析的過程
    比如從數據庫中取出幾條記錄需要顯示而模板文件只有可替換的一行變量的時候,就很需要這樣的功能。兩者都具有這樣的功能,只是使用時稍稍有些不同而已(PHPLIB Template採用parse(handler, value, true),而FastTemplate採用parse(handler, .value)在值的前面多加一個點),應該說PHPLIB Template的方法構造得相對優美一點。
  7. 區塊解析的過程(或者可以稱作動態解析)
    想像一下你需要從數據庫中取出符合條件的數據並顯示在網頁中--但是因爲條件會不盡相同,你並不能明確的知道會有多少條數據--這時候如果你又要採用模板,那麼區塊就是最好的選擇。它是在模板中用特定的符號定義的部分,這一部分可以反覆的被解析並添加到(而不是前一次的解析被後一次覆蓋)輸出網頁中。區塊也許就像下面顯示的一樣(左邊是PHPLIB Template採用的區塊設置,而右邊則是FastTemplate採用的):

好吧,如果你對以上蒼白的文字介紹還是有些摸不着頭腦,那麼我們就來看看兩個詳盡的模板處理的例程吧!(如果你有興趣對後面的測試代碼進行發掘,就會發現其實以下的兩個例子都來自那裏)


怎麼樣,是不是感覺幾乎是一致的?下面是區塊解析的例子,你也會發現同樣的效果:

我們的測試目標和結果
結束了對PHPLIB Template和FastTemplate的瞭解,應該可以進入本文的正題了--在應用環境中當然應該選擇易於使用同時速度理想的部件構建系統,那麼對於這樣的兩種類似技術,進行評測非常有必要。評測應該是由兩部分組成:技術的使用難度和速度的快慢程度--前者是評論的部分,而後者是測試的部分。對於前者,我們主要針對兩個類提供的API進行評論;對於後者,我們會讓測試的數據來說話,當然這中間免不了需要編寫一些簡單的測試代碼。

回合一:技術的易用性
這一回合主要是探討PHPLIB Template和FastTemplate提供的API的使用情況。應該說,前者提供的API更符合PHP的一些常見編碼慣例(特別是當你的項目中採用了PHPLIB的其他類時,這樣的規範性會對整個項目有好的影響);而後者的一些方法名總覺得有些彆扭(希望你不要覺得這只是我的狹隘看法,比如FastPrint()等等),同時方法的參數也不是非常"地道",這一點你也可以從剛纔的代碼看出來。

另外一點需要指出的是,對於模板區塊的解析,FastTemplate直到最近的版本纔開始支持。也就是說,如果你採用了之前的版本,在處理諸如數據庫中記錄的輸出等內容時,不得不把這塊內容獨立存儲在某處,然後在模板分析處理時附加上這個文件--真是一件讓人難受的事情,尤其是對網頁設計人員而言。

當然還有一點需要考察--那就是對於PHP版本的支持。PHPLIB產生在PHP3的時代,這一點和FastTemplate差不多;但是根據我們的應用,PHPLIB在現在的PHP4環境下運行相當好,而FastTemplate的網頁上則顯示了一些信息表明對於PHP4也許它還有一些BUG存在。

好了,講了這麼多(也許你會覺得都是FastTemplate的壞話),這個回合的勝利者很明顯:PHPLIB Template,尤其是你同時在使用PHPLIB的其他類時,這樣的技術易用性更加明顯(你將不會對這些出自同一個開發小組的API感到陌生)。

回合二:處理速度
也許這纔是很多人最關注的部分--在這個回合中,我們會採用兩種模板處理的方式:一種是常規的分析、替換,另一種是對區塊的解析、替換--同時這樣的兩種方式也是在實際系統中應用最多的:前者是一般的頁面處理,後者是關於數據庫內容的輸出處理。同時,由於兩種模板類採用的模板文件的格式基本相同,使得我們可以提供幾乎一致的模板文件分別供兩種模板解析,更增加了測試的可信度。

(有點複雜的測試方案)

首先是確定測試的硬件和軟件環境--硬件肯定是自己的機器了,Intel Celeron 733MHz, 256M RAM,40G HDD;軟件平臺中OS爲Win2K Pro,Web服務器爲Apache+PHP,且以模塊方式運行。

其次是規劃這次測試的系統--當然先在Web服務器的文檔根目錄下開一個tpl_test的新目錄用以放置這個測試的所有文件;然後在/tpl_test下建立include目錄以存放兩個模板類文件(我們測試的核心,以.inc.php爲文件擴展名)以及一個測試類文件(包括了計時和記錄日誌以及讀取日誌並分析等功能,以.inc.php爲文件擴展名)和一個數據文件(爲區塊解析的測試做準備,主要包含了一個二維數組,同樣以.inc.php爲文件擴展名),建立ihtml目錄存放使用的模板文件(需要被解析的模板文件,以.ihtml爲文件擴展名),建立logs目錄存放測試產生的日誌(後面就是發現,其實測試的數據就是由對這些日誌的分析得到的,以.log爲文件擴展名)。當然,兩種模板的處理PHP文件就放在/test目錄下。這次測試最關鍵的一點是,還需要建立一個PHP文件,對以上提到的負責模板處理的文件進行數次調用:比如一個文件fast_test.php是採用FastTemplate解析模板的,而phplib_test.php是採用PHPLIB Template解析的,那麼這個得出結果的PHP文件就負責多次以HTTP的方式請求以上的兩個頁面以獲得測試數據。

選擇待解析的模板和PHP程序編寫--因爲兩種模板處理方式對於模板文件本身的格式要求幾乎一致(比如待替換變量都採用{VAR}的形式等等),因此可以儘量保證同一測試中兩者選用的模板儘可能相同以謀求測試的最大公正性;同時在前文提到,爲模擬現實系統中常用的兩種模板應用:一般的頁面處理和對數據庫內容的輸出處理,測試使用的模板文件也分成兩種:一種是普通的帶有一些待替換變量的模板文件,另一種是帶有區塊的需要根據應輸出的內容反覆替換的模板文件。同樣對於這兩種模板文件,也需要分別編寫兩種不同的PHP文件進行解析。

測試方法--在瀏覽器中向/test/result.php提出請求,需要帶參數type=[simple|complex],在返回的結果中即可看到兩種模板在簡單或者複雜模式下的測試結果。

開展這樣的速度測試之前會擬定一個測試方案,簡單說來就是對於兩種處理方式分別編寫兩個PHP測試頁面,同時有一個控制測試的頁面多次調用這兩個頁面並記錄時間供採集測試數據。(如果有興趣你還可以參考以下詳細的測試方案,也許會對你深入瞭解這次測試有所幫助)

小結--在整個測試系統完成之後,我們應該能夠得到/test目錄中如下的文件清單:

Level 1 Level 2 Level 3 Remark
/test 測試系統的根目錄
result.php 進行測試併產生結果的PHP文件,測試時只需要在瀏覽器中請求該頁面即可獲得測試信息
simple__test_phplib.php 使用PHPLIB Template對一般模板進行分析的PHP文件
simple__test_fast.php 使用FastTemplate對一般模板進行分析的PHP文件
complex__test_phplib.php 使用PHPLIB Template對帶區塊模板進行分析的PHP文件
complex__test_fast.php 使用FastTemplate對帶區塊模板進行分析的PHP文件
/include 包含PHP類文件.inc.php
phplibTemplate.inc.php PHPLIB Template類文件
FastTemplate.inc.php FastTemplate類文件
TplTest.inc.php 測試中需要使用的測試類,包含諸如計時、讀取/分析日誌等方法。
data.inc.php 測試帶區塊模板時採用的數據文件。
/ihtml 包含模板文件.ihtml
simple_phplib.ihtml 採用PHPLIB Template處理的一般模板文件
simple_fast.ihtml 採用FastTemplate處理的一般模板文件
complex_phplib.ihtml 採用PHPLIB Template處理的帶區塊的模板文件
complex_fast.ihtml 採用FastTemplate處理的帶區塊的模板文件
/logs 包含日誌文件.log
simple_phplib.log 採用PHPLIB Template處理一般模板生成的日誌
simple_fast.log 採用FastTemplate處理一般模板生成的日誌
complex_phplib.log 採用PHPLIB Template處理帶區塊模板生成的日誌
complex_fast.log 採用FastTemplate處理帶區塊模板生成的日誌


經過了測試系統的設計和編寫,並且向負責網頁設計的同事討來兩個模板之後,我們就可以訪問這個系統了--前期的辛勤勞動使得現在觀看結果的工作只需要在瀏覽器的地址欄中打入 http://localhost/tpl_test/result.php?type=[simple|complex] (如果你是在其他的非本地服務器中進行這個測試,那麼域名應採用所在服務器的域名--比如我自己的機器叫做patrick等等)。下面是我自己在某一次的測試中獲得的結果:(測試結果數據解釋)

名稱 解釋 備註
amount 測試總數(連續請求該頁面總數) 該參數可在result.php文件中修改
max_seq 最大處理時間的序號 範圍在1-amount之間
max_value 最大處理時間的值 峯值數據供參考
min_seq 最小處理時間的序號 範圍在1-amount之間
min_value 最小處理時間的值 峯值數據供參考
average 平均處理時間 測試中最有價值的數據


當然,如果你覺得一次測試的結果並不可靠,可以反覆按下瀏覽器的刷新按鈕,就能夠觀察到不同測試的結果(理論上應該是相差無幾)。

測試結果以及頒發"XX選擇獎"
好了,在偏重速度測試的回合二中PHPLIB Template以驚人的2倍的速度戰勝了FastTemplate;而同時在第一回閤中PHPLIB Template有以良好的API設計和易用性佔得上風。結果顯而易見--我們的選擇獎當然頒發給了PHPLIB Template,同時這次的測試也讓我們對PHPLIB這個類庫設計有了更深的瞭解。

主觀評價
既然有了結果,那麼FastTemplate自然也就不能進入我們的項目了--雖然從結果上看來我們花費了半天的時間得到了一個毫無變化的結果(PHPLIB Template繼續很好的在項目中使用),但是測試的過程卻是很有價值的,特別是採用PHP進行測試的方法,應該會在以後的類似決策中起到一定參考作用。

參考資料
本文中的測試代碼下載 (test_source.zip)

介紹PHPLIB Template和FastTemplate的著名文章

  1. Templates, The PHPLIB Way
    by David Orr, May of 2000
    (http://www.phpbuilder.com/columns/david20000512.php3 )
    (PHPLIB Template)
  2. Templates - why and how to use them in PHP3
    by Sascha Schumann, March of 1999
    (http://www.phpbuilder.com/columns/sascha19990316.php3 )
    (FastTemplate)

PHPLIB Template和FastTemplate的主頁

  1. PHP Library Homepage
    (http://phplib.netuse.de/ )
  2. FastTemplate Homepage
    (http://www.thewebmasters.net/php/FastTemplate.phtml )

其他模板處理文章以及其他關於PHP的資料
PHP Builder (http://www.phpbuilder.com/ )

以上文章的中文譯本以及其他的中文PHP資料
PHP中文用戶(http://phpuser.com/ )

關於作者
王晨,一名Web開發者,同時感興趣於撰寫和翻譯與自己的技術愛好相關的文章(在工作和發表文章的時候更喜歡別人叫自己Patrick)。本人的主要開發經歷集中於PHP,現在的愛好更多的在於Java;同時也是Open Source的擁護者之一。閒暇時也經常喜歡去書店逛逛;經常活動於家附近以及母校附近--畢業於東南大學。您可以通過電子郵件 [email protected] 跟他聯繫。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章