PHP以其易用性得到迅速的推廣,但易用並不是說就能用好它,實際上許多程式員用它很容易的立一個個WEB應用系統,但又有多少人仔細的考慮過他們的代碼,是否容易維護、是否足夠健壯、否效率足夠高、是否足夠安全,當PHP用於建立大型網站時這些就成為很關鍵的因素。下面我們從較輕微的問題開始討論,直至一些致命的錯誤。共分三部分。
第一部分、較輕微的錯誤
一、Printf(),
該函數主要用來格式化顯示資料。當你要改變某個資料的顯示格式時才使用。
例如以不同的精度來顯示PI(3.1415926)的值。
<?php
/*
* The three faces of Π
*/
printf ("Pi is: %.2f\n<br>\n", M_PI);
printf ("Pi is also: %.3f\n<br>\n", M_PI);
printf ("Pi is also: %.4f\n<br>\n", M_PI);
?>
但許多程式員僅僅為顯示一些變數值和函數傳回值使用該函數。因為Printf()在顯示資料前要先格式化該資料以速度較慢,因此,僅為了顯示資料時應用print和echo,以提高速度。
二、語意檢查
PHP是一種弱類型語言,也就是說在使用一個變數前不用定義,這樣給編程帶來了很大的方便和靈活,但你自己必須知道該變數到底應該是哪種類型,因為該變數在運行時仍實際對應著某一種類型(各種類型之間可以自由互相轉換),沒有類型的變數是不存在的。有可能PHP並不能檢查出你的語意錯誤,但由於變數類型的變化,會導致一些潛在的問題的發生。另外一個值得注意的問題是變數的範圍,它也可能會導致一些潛在的問題的發生。
在PHP中有以下幾種基本變數:
Boolean, resource, integer, double, string, array and object。
三、臨時變數的使用
臨時變數的濫用會導致程式運行效率的降低。何時使用臨時變數可基於以下兩點考慮:
1、該變數是否至少使用兩次。
2、該變數的使用是否會顯著提高程式的可讀性。
如果一條也不滿足,則省略該變數的使用。例如:
<?php
$tmp = date ("F d, h:i a"); /* ie January 3, 2:30 pm */
print $tmp;
?>
就應該改成:
<?php
print date ("F d, h:i a");
?>
又如:
<?php
// string reverse_characters(string str)
// Reverse all of the characters in a string.
function reverse_characters ($str)
{
return implode ("", array_reverse (preg_split("//", $str)));
}
?>
的可讀性不強,可改成:
<?php
// string reverse_characters(string str)
// Reverse all of the characters in a string.
function reverse_characters ($str)
{
$characters = preg_split ("//", $str);
$characters = array_reverse ($characters);
return implode ("", $characters);
}
?>
四、用戶端和伺服器端代碼的分離
用戶端和伺服器端代碼的在PHP程式中實際上就是HTML代碼和PHP語言代碼,很多人把HTML和PHP語句混合在一個檔案裡,使得這檔案很大,這種風格對程式的維護和再開發很不利,不適合大型網站的開發。一般有兩種方法把HTML和PHP語句分開:
1、編寫專用API,例如:
index.php ? The Client side
<?php include_once ("site.lib"); ?>
<html>
<head>
<title> <?php print_header (); ?> </title>
</head>
<body>
<h1> <?php print_header (); ?> </h1>
<table border="0" cellpadding="0" cellspacing="0">
<tr>
<td width="25%">
<?php print_links (); ?>
</td>
<td>
<?php print_body (); ?>
</td>
</tr>
</table>
</body>
</html>
site.lib ? The server side code
<?php
$dbh = mysql_connect ("localhost", "sh", "pass")
or die (sprintf ("Cannot connect to MySQL [%s]: %s",
mysql_errno (), mysql_error ()));
@mysql_select_db ("MainSite")
or die (sprintf ("Cannot select database [%s]: %s",
mysql_errno (), mysql_error ()));
$sth = @mysql_query ("SELECT * FROM site", $dbh)
or die (sprintf ("Cannot execute query [%s]: %s",
mysql_errno (), mysql_error ()));
$site_info = mysql_fetch_object ($sth);
function print_header ()
{
global $site_info;
print $site_info->header;
}
function print_body ()
{
global $site_info;
print nl2br ($site_info->body);
}
function print_links ()
{
global $site_info;
$links = explode ("\n", $site_info->links);
$names = explode ("\n", $site_info->link_names);
for ($i = 0; $i < count ($links); $i++)
{
print "\t\t\t
<a href=\"$links[$i]\">$names[$i]</a>
\n<br>\n";
}
}
?>
這種方法使得程式看起來比較簡潔,而且執行速度也較快。
2、使用模板的方法
這種方法使得程式看起來更簡潔,同樣實現上面的功能,可用以下代碼:
<html>
<head>
<title>%%PAGE_TITLE%%</title>
</head>
<body %%BODY_PROPERTIES%%>
<h1>%%PAGE_TITLE%%</h1>
<table border="0" cellpadding="0" cellspacing="0">
<tr>
<td width="25%">%%PAGE_LINKS%%</td>
<td>%%PAGE_CONTENT%%</td>
</tr>
</table>
</body>
</html>
用預留位置代替要動態產生的內容,然後用一解析程式分析該模板檔案,把預留位置用際的內容替換。種方法使得即使不會使用PHP的頁面製作人員也能修改模板檔案。這種方法的缺點是執行效率不高,因為要解釋模板檔案。同時實現起來也比較複雜。
註: www.thewebmasters.net的 FastTemplate class可方便的實現以上功能。
五、不要用過時的函數
作為一種自由軟體,PHP發展很快,其中的很多函數都已淘汰,例如:
while (1):
print "5";
if ($idx++ == 5):
break;
endif;
endwhile;
雖然還能用但效率肯定不高,而且可能在以後的版本中會禁用,導致程式不能運行。因此要經常對照最新PHP手冊檢查那些函數已淘汰及時修正。