二月11

撰寫 PHP 程式的一些優化建議

這裡有一篇 Tsung 翻譯的「PHP 程式效能優化的 40 條建議」,覺得很不錯。個人接觸 PHP 這個語言約有四個年頭了,再撰寫時多少會注意一些效能細節,不過有時為了讓程式碼更好看、簡短,往往就忽略掉這些細節,這篇的建議正好可以讓新手學習、老手複習一些 PHP 的最佳化技巧。
引用內容:

如果一個方法可靜態化,就對它做靜態聲明。速率可提升至4倍
echo 比 print 快
使用echo的多重參數(譯註:指用逗號而不是句點)代替字元串連接
在執行for循環之前確定最大循環數,不要每循環一次都計算最大值
註銷那些不用的變量尤其是大數組,以便釋放內存
盡量避免使用__get,__set,__autoload
require_once()代價昂貴
在包含文件時使用完整路徑,解析操作系統路徑所需的時間會更少
如果你想知道腳本開始執行(譯註:即服務器端收到客戶端請求)的時刻,使用$_SERVER['REQUEST_TIME']要好於time()
函數代替正則表達式完成相同功能
str_replace函數比preg_replace函數快,但strtr函數的效率是str_replace函數的四倍
如果一個字元串替換函數,可接受數組或字元作為參數,並且參數長度不太長,那麼可以考慮額外寫一段替換程式,使得每次傳遞參數是一個字元,而不是只寫一行程式接受數組作為查詢和替換的參數
使用選擇分支語句(譯註:即switch case)好於使用多個if,else if語句
用@屏蔽錯誤消息的做法非常慢
打開apache的mod_deflate模塊
資料庫連接當使用完畢時應關掉
$row['id']的效率是$row[id]的7倍
錯誤訊息代價昂貴
盡量不要在for循環中使用函數,比如for ($x=0; $x < count($array); $x)每循環一次都會調用count()函數
在function中遞增局部變量,速度是最快的。幾乎與在函數中調用局部變量的速度相當
遞增一個全局變量要比遞增一個局部變量慢2倍
遞增一個物件屬性(如:$this->prop++)要比遞增一個局部變量慢3倍
遞增一個未預定義的局部變量要比遞增一個預定義的局部變量慢9至10倍
僅定義一個局部變量而沒在函數中調用它,同樣會減慢速度(其程度相當於遞增一個局部變量)。PHP大概會檢查看是否存在全局變量
方法調用看來與類中定義的方法的數量無關,因為我(在測試方法之前和之後都)添加了10個方法,但性能上沒有變化
派生類中的方法運行起來要快於在基類中定義的同樣的方法
調用帶有一個參數的空函數,其花費的時間相當於執行7至8次的局部變量遞增操作。類似的方法調用所花費的時間接近於15次的局部變量遞增操作
用單引號代替雙引號來包含字元串,這樣做會更快一些。因為PHP會在雙引號包圍的字元串中搜尋變量,單引號則不會。當然,只有當你不需要在字元串中包含變量時才可以這麼做
輸出多個字元串時,用逗號代替句點來分隔字元串,速度更快。註意:只有echo能這麼做,它是一種可以把多個字元串當作參數的“函數”(譯註:PHP手冊中說echo是語言結構,不是真正的函數,故把函數加上了雙引號)
Apache解析一個PHP腳本的時間要比解析一個靜態HTML頁面慢2至10倍。盡量多用靜態HTML頁面,少用腳本
除非腳本可以緩存,否則每次調用時都會重新編譯一次。引入一套PHP緩存機制通常可以提升25%至100%的性能,以免除編譯開銷
盡量做緩存,可使用memcached。memcached是一款高性能的內存物件緩存系統,可用來加速動態Web應用程式,減輕數據庫負載。對運算碼 (OP code)的緩存很有用,使得腳本不必為每個請求做重新編譯
當操作字元串並需要檢驗其長度是否滿足某種要求時,你想當然地會使用strlen()函數。此函數執行起來相當快,因為它不做任何計算,只返回在zval 結構(C的內置數據結構,用於存儲PHP變量)中存儲的已知字元串長度。但是,由於strlen()是函數,多多少少會有些慢,因為函數調用會經過諸多步驟,如字母小寫化(譯註:指函數名小寫化,PHP不區分函數名大小寫),會跟隨被調用的函數一起執行。在某些情況下,你可以使用isset() 技巧加速執行你的程式舉例如下:if (strlen($foo) < 5) { echo “Foo is too short”; }與下面的技巧做比較:if (!isset($foo{5})) { echo “Foo is too short”; }調用isset()恰巧比strlen()快,因為與後者不同的是,isset()作為一種語言結構,意味著它的執行不需要函數查找和字母小寫化。也就是說,實際上在檢驗字元串長度的頂層程式中你沒有花太多開銷
當執行變數$i的遞增或遞減時,$i++會比++$i慢一些。這種差異是PHP特有的,並不適用於其他語言,所以請不要修改你的C或Java程式並指望它們能立即變快,沒用的。++$i更快是因為它只需要3條指令(opcodes),$i++則需要4條指令。後置遞增實際上會產生一個臨時變量,這個臨時變量隨後被遞增。而前置遞增直接在原值上遞增。這是最優化處理的一種,正如Zend的PHP優化器所作的那樣。牢記這個優化處理不失為一個好主意,因為並不是所有的指令優化器都會做同樣的優化處理,並且存在大量沒有裝配指令優化器的網際網路服務提供商(ISPs)和服務器
並不是所有都要使用OOP,面向物件往往開銷很大,每個方法和物件調用都會消耗很多Memory
並非要用 class 實現所有的數據結構,數組也很有用
不要把方法細分得過多,仔細想想你真正打算重用的是哪些程式?
當你需要時,你總能把程式分解成方法
盡量採用大量的PHP內置函數
如果在程式中存在大量耗時的函數,你可以考慮用C擴展的方式實現它們
評估檢驗(profile)你的程式。檢驗器會告訴你,程式的哪些部分消耗了多少時間。Xdebug調試器包含了檢驗程式,評估檢驗總體上可以顯示出程式的瓶頸
mod_zip可作為Apache模塊,用來即時壓縮你的數據,並可讓數據傳輸量降低80%
另一篇優化PHP的精彩文章,由John Lim撰寫


十二月22

CodeIgniter 設定 MySQL 編碼

※ CodeIgniter 1.6 已釋出,不需再做這篇的修正了哦!
先前有提到過 CodeIgniter 這個 PHP Framework,也用它做了幾個 case,當然開發速度不能說快上多少,但確實讓程式碼變得整齊優雅許多。
使用 CI 目前遇到第一個問題是存取 MySQL 資料的問題,在 MySQL 4.1 以後採用了多層式編碼設定,簡單說就是不管是資料庫、資料表、欄位或是 Client 連線階段,都可以分別設定編碼,所以即使 server 都改成了 utf8,PHP 到 MySQL Server 的這個 connection 卻還是預設的(latin1),許多亂碼就是這樣造成的,應該已經有不少人知道解決方法,就是先送出一個改變編碼的查詢:SET NAMES 'charset'。
然而目前 CodeIgniter 並沒有這樣的設計,所以只好自行改 code 囉!
Step1:修改 system/application/config/database.php
增加一項:$db['default']['charset'] = "utf8";
Step2:修改 system/database/DB_driver.php
在 32 行之後的宣告增加:var $charset = '';
接著 96 行的陣列增加:'charset' => '',
Step3:修改 system/database/drivers/mysql/mysql_driver.php
74 行 [...]

十月5

建立網頁縮圖(Webthumb)

目前不少網站都有自動產生網頁縮圖的功能,像是 HEMiDEMi、funP 推推王、Alexa 等等,雖然很想知道其中所使用的技術,不過似乎很難找到相關文章,大多是倚賴提供縮圖服務的網站居多。
Bluga.net WebThumb
我覺得很不錯的縮圖服務網站,每個月可免費建立 250 組縮圖,且提供 API 可以應用在程式上,產生縮圖速度算滿快的,大約 2~5 分鐘即可顯示。
artViper Website Tools
這網站有提供一些 Website Tools,其中就有網頁縮圖的功能。
Website Thumbnail Generator
這個是不需要依賴其他網站的,使用了 IECapt 這個小工具,所以主機只能是 Windows,配合 PHP 來自動產生縮圖。
延伸閱讀:
Website Thumbnail Creators (Website Snapshot Generators)

十月2

用 PHP 寫 RS232 串列通訊程式

相信經常在翻 PHP 手冊的人應該都有察覺,PHP 不只是可以用來開發網頁程式而已,很多其他語言能完成的工作 PHP 也都做的到,例如透過 PHP 的 Direct IO Functions 就可以對 Serial port 溝通,而且可以在 Windows 下執行,一般在 Windows 要實做 com port 通訊程式,大多是透過像 VB/VC++/C++ Builder 這類程式所提供的元件來撰寫,不過用 PHP 就可以很快實做出一般的功能囉!
在開始之前要先讓 PHP 支援 dio 才行,5.1.0 版之後已經將 dio 移至 PECL 了,可以到 PHP 官網下載編譯好的 PECL,解壓縮後將裡面的 php_dio.dll 放到 PHP 目錄的 ext 底下,接著修改 php.ini 加入 extension=php_dio.dll 即可。
在 Windows 底下讀取 COM1 的例子:
<?phpexec(’mode COM1: baud=19200 [...]

八月3

JavaScript 讀取外部文字檔

想要利用 JavaScript 來讀取一個檔案內容,搜尋到的方法很多都是用 ActiveX 物件來達成,這樣缺點就只能 IE Only 了。不過後來發現其實用 Ajax 就可以達成了。
搭配 JQuery 使用:
<script type="text/javascript">$(document).ready(function() {    $.get(’content.txt’, function(data){        alert(data);    });});</script>
這裡透過 Ajax 發出 GET Request 到 content.txt,就會直接將檔案的內容傳回輸出,很簡單吧^^
但是由於 Ajax 考慮到安全性,所以只能作用在同網域底下。如果要讀的檔案是遠端主機上的呢?這時可以利用其他程式當作中間人,例如透過 PHP 去取得不同網域下的檔案。
JS 內容:
$(document).ready(function() {    $.get(’readfile.php’, function(data){        alert(data);    });});
PHP 內容:
<?phpecho file_get_contents("http://aaa.com.tw/content.txt");?>