PHP是一種執行起來非常迅速的編程語言,但是比起僅僅優化代碼來說仍然值得優化PHP本身。 本文我們將根據一些實效闡述為什麼優化PHP本身要比優化代碼來的更貼切,以及為什麼需要理解根據PHP在你的服務器上其他相關子系統的表現找出瓶頸並修復之。與此同時,我們也提到了如何優化您的PHP代碼來讓他們擁有更快的執行速度。
獲得高性能 當我們談及好的性能,往往不僅僅是指您的PHP代碼執行起來有多快。性能是一套在可量化評測和速度之間取出的平衡。僅僅依靠使用更少資源的代碼執行起來也許比在高速緩存中之行的代碼更慢,並且相同的一組(在高速緩沖中執行的)代碼可以在同時並發執行在一台Web服務器上。 在下面的例子中,A.php算作一位盡可能跑得快的賽跑選手,而B.php是一個幾乎可以以同一慢速永遠跑下去的馬拉松選手。輕負荷情況下,A.php可以充分的快,但是當流量增加後,B.php的性能表現將僅僅降低一點點而A.php會垮掉。 讓我們來通過一個事實來驗證此說法更深遠的本質意義。假設我們需要讀取一個250K的文件並生成一個關於此文件的Html概要。我們寫了兩個腳本來做同樣一件事:hare.php將一次性讀取整個文件到內存中,然後一步執行到位;而tortoise.php每次只讀取文件的一行,並且決不超過內容容量。結果Tortoise.php因為多次讀寫需要更多的系統回應而慢得多。 程序每執行一次,hare.php需要0.04秒CPU執行時間和10Mb的內存,而tortoise.php需要0.06秒CPU執行時間和5Mb的內存。服務器共有100Mb實際內存容量並且其CPU有99%是空閒的。我們同時假定執行這樣一個簡單事件不產生內存碎片。 當有10各程序並發執行時,hare.php將發生內存溢出(10 ×10 = 100)。與此同時,tortoise.php仍將有50Mb空余內存可用!11個程序並發執行將使hare.php徹底“潰敗”因為它開始需要使用虛擬內存——執行速度有可能降低到其常規速度的一半以下;而且現在每一個單獨程序進程需要0.08秒CPU執行時間。而此期間,tortoise.php仍舊運行在其常規CPU執行時間——0.06秒! 以下表格中,執行得更快的PHP腳本使用粗體區分開來: ------------------------------------------------------------------------------------------ 連接數 每執行1次HTTP請求所需CPU執行時間(秒) 每執行10次HTTP請求所需CPU執行時間(秒) 每執行11次HTTP請求所需CPU執行時間(秒) ------------------------------------------------------------------------------------------ hare.php 0.04 0.04 0.88(內存溢出) ------------------------------------------------------------------------------------------ tortoise.php 0.06 0.60 0.66 ------------------------------------------------------------------------------------------ 如您在上例中看到的,獲得更好的性能不再僅僅是寫出執行起來更快的PHP程序。高性能PHP表現需要對底層硬件知識以及操作系統、軟件支持如Web服務器、數據庫等有一個良好認識和理解。
瓶頸 以上兩個例子讓我們看到了(性能)下降的瓶頸所在。當擁有無限大容量的內存時,hare.php的確是始終比tortoise.php快。但是,僅僅認為內存是PHP整體性能的瓶頸所在顯得過於單純——實際上遠不止這些: (a) 網絡 你的網絡有可能是最大的瓶頸所在。如果你有10M的帶寬——最多你只能獲得1M/秒的傳輸速度。如果假設每個PHP頁為30k,那麼每秒僅僅只傳輸33頁就將使你的網絡帶寬達到飽和。更多導致瓶頸產生的因素包括頻繁訪問低速DNS,或者網絡設備僅能獲得十分有限的存儲。 (b) CPU 如果你監視一下你的CPU負荷情況,發送一個純靜態HTML頁面並不會增加CPU負擔——就像我們以上提到的,此時瓶頸在於網絡。當然啦,對於由PHP生成的復雜動態頁面,你的CPU速度自然將成為限制因素之一。擁有包含多個CPU的服務器或者一個服務器陣列將減輕因CPU帶來的影響。 (c) 共享存儲 共享存儲主要在通訊進程中發揮作用,或者是用於儲存公共資源(比如多個CPU間的緩存數據或代碼)。如果共享存儲分配不足,同樣將導致瓶頸的發生。 (d) 文件系統 訪問硬盤的速度至少比直接讀取內存要慢50到100倍。使用高速緩存將使情況有所好轉。然而低內存空間將適用於文件緩沖的內存(通常稱為虛擬內存)空間減少,進而導致整體性能下降。虛擬內存容易產生大量文件碎片,導致訪問硬盤速度下降。在Unix系統上使用大量鏈接標記同樣也會石訪問硬盤的速度下降。缺省Linux安裝模式下的默認硬盤訪問設置也是很不被看好的,因為它是為了兼容性而不是速度。必要時請使用hdparm命令來更改Linux硬盤配置。 (e) 進程管理 在某些操作系統上(比如說Windows系統)創建一個新的進程是很緩慢的操作。這意味著在這些操作系統上作涉及到進程創建的CGI應用會慢許多。這種時候在多進程模式下運行PHP將會有所改善。(注意:老版本PHP尚不支持多進程模式) 應避免你的服務器產生過多的無效進程。舉個例子,如果你的服務器是純Web服務,需關閉(設置是不安裝)X-Windows窗口系統。Windows系統上,應禁用微軟快速搜索(Office的一部分)和第三方屏保程序(通常會需要100%的CPU利用)。 有些程序你可以考慮刪除包括一些不常用的網絡協議、郵件服務、病毒掃描、(鼠標,紅外線端口)驅動程序等。Unix系統上,假定你通過SSH訪問你的服務器,那麼你可以考慮去除: telnetd,inetd,atd,FTPd,lpd,sambad模塊 用於收取郵件的sendmail模塊 NFS的映射表 xfs,fvwm,xinit,X 你也可以通過修改啟動腳本(其路徑通常位於/etc/init* 或 /etc/rc*/init*)來實現禁用不同的啟動程序。 也可以檢查一下定期處理任務安排是否合理,看是否可以在服務器空閒期執行。 (f) 連接到其他服務器 如果你的服務器需要其他服務器提供的服務,此時有可能是其他服務器產生了瓶頸。最常見的就是一個很慢的執行諸多復雜查詢的數據庫服務器服務於很多其他服務器。