本文介紹一些關於改善和優化PHP代碼的提示和技巧,供大家參考,具體內容如下
1.不要使用相對路徑,要定義一個根路徑
這樣的代碼行很常見:
require_once('../../lib/some_class.php');
這種方法有很多缺點:
1)、它首先搜索php包括路徑中的指定目錄,然後查看當前目錄。因此,會檢查許多目錄。
2)、當一個腳本被包含在另一個腳本的不同目錄中時,它的基本目錄變為包含腳本的目錄。
3)、另一個問題是,當一個腳本從cron運行時,它可能不會將它的父目錄作為工作目錄。
所以使用絕對路徑便成為了一個好方法:
define('ROOT' , '/var/www/project/'); require_once(ROOT . '../../lib/some_class.php'); //rest of the code
這就是一個絕對路徑,並且會一直保持不變。但是,我們可以進一步改善。目錄/var/www/project可以變,那麼我們每次都要改嗎?
不,使用魔術常量如__FILE__可以讓它變得可移植。請仔細看:
//suppose your script is /var/www/project/index.php //Then __FILE__ will always have that full path. define('ROOT' , pathinfo(__FILE__, PATHINFO_DIRNAME)); require_once(ROOT . '../../lib/some_class.php'); //rest of the code
所以現在,即使你將項目轉移到一個不同的目錄,例如將其移動到一個在線的服務器上,這些代碼不需要更改就可以運行。
2.不使用require,包括require_once或include_once
你的腳本上可能會包括各種文件,如類庫,實用程序文件和輔助函數等,就像這些:
require_once('lib/Database.php'); require_once('lib/Mail.php'); require_once('helpers/utitlity_functions.php');
這相當粗糙。代碼需要更加靈活。寫好輔助函數可以更容易地包含東西。舉個例子:
function load_class($class_name) { //path to the class file $path = ROOT . '/lib/' . $class_name . '.php'); require_once( $path ); } load_class('Database'); load_class('Mail');
看到區別了嗎?很明顯。不需要任何更多的解釋。
你還可以進一步改善:
function load_class($class_name) { //path to the class file $path = ROOT . '/lib/' . $class_name . '.php'); if(file_exists($path)) { require_once( $path ); } }
這樣做可以完成很多事情:
為同一個類文件搜索多個目錄。
輕松更改包含類文件的目錄,而不破壞任何地方的代碼。
使用類似的函數用於加載包含輔助函數、HTML內容等的文件。
3.在應用程序中維護調試環境
在開發過程中,我們echo數據庫查詢,轉儲創造問題的變量,然後一旦問題被解決,我們注釋它們或刪除它們。但讓一切留在原地可提供長效幫助。
在開發計算機上,你可以這樣做:
define('ENVIRONMENT' , 'development'); if(! $db->query( $query ) { if(ENVIRONMENT == 'development') { echo "$query failed"; } else { echo "Database error. Please contact administrator"; } }
並且在服務器上,你可以這樣做:
define('ENVIRONMENT' , 'production'); if(! $db->query( $query ) { if(ENVIRONMENT == 'development') { echo "$query failed"; } else { echo "Database error. Please contact administrator"; } }
4.通過會話傳播狀態消息
狀態消息是那些執行任務後生成的消息。
<?php if($wrong_username || $wrong_password) { $msg = 'Invalid username or password'; } ?> <html> <body> <?php echo $msg; ?> <form> ... </form> </body> </html>
這樣的代碼很常見。使用變量來顯示狀態信息有一定的局限性。因為它們無法通過重定向發送(除非你將它們作為GET變量傳播給下一個腳本,但這非常愚蠢)。而且在大型腳本中可能會有多個消息等。
最好的辦法是使用會話來傳播(即使是在同一頁面上)。想要這樣做的話在每個頁面上必須得有一個session_start。
function set_flash($msg) { $_SESSION['message'] = $msg; } function get_flash() { $msg = $_SESSION['message']; unset($_SESSION['message']); return $msg; }
在你的腳本中:
<?php if($wrong_username || $wrong_password) { set_flash('Invalid username or password'); } ?> <html> <body> Status is : <?php echo get_flash(); ?> <form> ... </form> </body> </html>
5.讓函數變得靈活
function add_to_cart($item_id , $qty) { $_SESSION['cart'][$item_id] = $qty; } add_to_cart( 'IPHONE3' , 2 );
當添加單一條目時,使用上面的函數。那麼當添加多個條目時,就得創建另一個函數嗎?NO。只要讓函數變得靈活起來使之能夠接受不同的參數即可。請看:
function add_to_cart($item_id , $qty) { if(!is_array($item_id)) { $_SESSION['cart'][$item_id] = $qty; } else { foreach($item_id as $i_id => $qty) { $_SESSION['cart'][$i_id] = $qty; } } } add_to_cart( 'IPHONE3' , 2 ); add_to_cart( array('IPHONE3' => 2 , 'IPAD' => 5) );
好了,現在同樣的函數就可以接受不同類型的輸出了。以上代碼可以應用到很多地方讓你的代碼更加靈活。
6.省略結束的php標簽,如果它是腳本中的最後一行
我不知道為什麼很多博客文章在談論php小技巧時要省略這個技巧。
<?php echo "Hello"; //Now dont close this tag
這可以幫助你省略大量問題。舉一個例子:
類文件super_class.php
<?php class super_class { function super_function() { //super code } } ?> //super extra character after the closing tag
現在看index.php
require_once('super_class.php'); //echo an image or pdf , or set the cookies or session data
你會得到發送錯誤的Header。為什麼呢?因為“超級多余字符”,所有標題都去處理這個去了。於是你得開始調試。你可能需要浪費很多時間來尋找超級額外的空間。
因此要養成省略結束標簽的習慣:
<?php class super_class { function super_function() { //super code } } //No closing tag
這樣更好。
7.在一個地方收集所有輸出,然後一次性輸出給浏覽器
這就是所謂的輸出緩沖。比方說,你從不同的函數得到像這樣的內容:
function print_header() { echo "<div id='header'>Site Log and Login links</div>"; } function print_footer() { echo "<div id='footer'>Site was made by me</div>"; } print_header(); for($i = 0 ; $i < 100; $i++) { echo "I is : $i <br />'; } print_footer();
其實你應該先在一個地方收集所有輸出。你可以要麼將它存儲於函數中的變量內部,要麼使用ob_start和ob_end_clean。所以,現在應該看起來像這樣
function print_header() { $o = "<div id='header'>Site Log and Login links</div>"; return $o; } function print_footer() { $o = "<div id='footer'>Site was made by me</div>"; return $o; } echo print_header(); for($i = 0 ; $i < 100; $i++) { echo "I is : $i <br />'; } echo print_footer();
那麼,為什麼你應該做輸出緩沖呢:
你可以在將輸出發送給浏覽器之前更改它,如果你需要的話。例如做一些str_replaces,或者preg_replaces,又或者是在末尾添加一些額外的html,例如profiler/debugger輸出。
發送輸出給浏覽器,並在同一時間做php處理並不是好主意。你見過這樣的網站,它有一個Fatal error在側邊欄或在屏幕中間的方框中嗎?你知道為什麼會出現這種情況嗎?因為處理過程和輸出被混合在了一起。
8.當輸出非HTML內容時,通過header發送正確的mime類型
請看一些XML。
$xml = '<?xml version="1.0" encoding="utf-8" standalone="yes"?>'; $xml = "<response> <code>0</code> </response>"; //Send xml data echo $xml;
工作正常。但它需要一些改進。
$xml = '<?xml version="1.0" encoding="utf-8" standalone="yes"?>'; $xml = "<response> <code>0</code> </response>"; //Send xml data header("content-type: text/xml"); echo $xml;
請注意header行。這行代碼告訴浏覽器這個內容是XML內容。因此,浏覽器能夠正確地處理它。許多JavaScript庫也都依賴於header信息。
JavaScript,css,jpg圖片,png圖像也是一樣:
JavaScript
header("content-type: application/x-javascript"); echo "var a = 10"; CSS header("content-type: text/css"); echo "#div id { background:#000; }"
9.為MySQL連接設置正確的字符編碼
曾碰到過unicode/utf-8字符被正確地存儲在mysql表的問題,phpmyadmin也顯示它們是正確的,但是當你使用的時候,你的網頁上卻並不能正確地顯示。裡面的奧妙在於MySQL連接校對。
$host = 'localhost'; $username = 'root'; $password = 'super_secret'; //Attempt to connect to database $c = mysqli_connect($host , $username, $password); //Check connection validity if (!$c) { die ("Could not connect to the database host: <br />". mysqli_connect_error()); } //Set the character set of the connection if(!mysqli_set_charset ( $c , 'UTF8' )) { die('mysqli_set_charset() failed'); }
一旦你連接到數據庫,不妨設置連接字符集。當你在你的應用程序中使用多種語言時,這絕對有必要。
否則會發生什麼呢?你會在非英文文本中看到很多的方框和????????。
10.使用帶有正確字符集選項的htmlentities
PHP 5.4之前,使用的默認字符編碼是ISO-8859-1,這不能顯示例如À â 這樣的字符。
$value = htmlentities($this->value , ENT_QUOTES , 'UTF-8');
從PHP 5.4起,默認編碼成了UTF-8,這解決了大部分的問題,但你最好還是知道這件事,如果你的應用程序使用多種語言的話。
先介紹這10個技巧,剩下的PHP技巧我們將在接下來的文章中為大家分享,感謝您的閱讀。