摘要:,,圖片,圖像也是一樣為連接設置正確的字符編碼曾碰到過字符被正確地存儲在表的問題,也顯示它們是正確的,但是當你使用的時候,你的網頁上卻并不能正確地顯示。使用帶有正確字符集選項的之前,使用的默認字符編碼是,這不能顯示例如這樣的字符。
今天我們要介紹一些關于改善和優化PHP代碼的提示和技巧。請注意,這些PHP技巧適用于初學者,而不是那些已經在使用MVC框架的人。
1.不要使用相對路徑,要定義一個根路徑這樣的代碼行很常見:
require_once("../../lib/some_class.php");
這種方法有很多缺點:
它首先搜索php包括路徑中的指定目錄,然后查看當前目錄。因此,會檢查許多目錄。
當一個腳本被包含在另一個腳本的不同目錄中時,它的基本目錄變為包含腳本的目錄。
另一個問題是,當一個腳本從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.通過會話傳播狀態消息
狀態消息是那些執行任務后生成的消息。
這樣的代碼很常見。使用變量來顯示狀態信息有一定的局限性。因為它們無法通過重定向發送(除非你將它們作為GET變量傳播給下一個腳本,但這非常愚蠢)。而且在大型腳本中可能會有多個消息等。
最好的辦法是使用會話來傳播(即使是在同一頁面上)。想要這樣做的話在每個頁面上必須得有一個session_start。
function set_flash($msg) { $_SESSION["message"] = $msg; } function get_flash() { $msg = $_SESSION["message"]; unset($_SESSION["message"]); return $msg; }
在你的腳本中:
Status is :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小技巧時要省略這個技巧。
這可以幫助你省略大量問題。舉一個例子:
類文件super_class.php
//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。為什么呢?因為“超級多余字符”,所有標題都去處理這個去了。于是你得開始調試。你可能需要浪費很多時間來尋找超級額外的空間。
因此要養成省略結束標簽的習慣:
這樣更好。
7.在一個地方收集所有輸出,然后一次性輸出給瀏覽器這就是所謂的輸出緩沖。比方說,你從不同的函數得到像這樣的內容:
function print_header() { echo "Site Log and Login links"; } function print_footer() { echo " "; } print_header(); for($i = 0 ; $i < 100; $i++) { echo "I is : $i
"; } print_footer();其實你應該先在一個地方收集所有輸出。你可以要么將它存儲于函數中的變量內部,要么使用ob_start和ob_end_clean。所以,現在應該看起來像這樣
function print_header() { $o = "Site Log and Login links"; return $o; } function print_footer() { $o = " "; return $o; } echo print_header(); for($i = 0 ; $i < 100; $i++) { echo "I is : $i
"; } echo print_footer();那么,為什么你應該做輸出緩沖呢:
你可以在將輸出發送給瀏覽器之前更改它,如果你需要的話。例如做一些str_replaces,或者preg_replaces,又或者是在末尾添加一些額外的html,例如profiler/debugger輸出。
8.當輸出非HTML內容時,通過header發送正確的mime類型
發送輸出給瀏覽器,并在同一時間做php處理并不是好主意。你見過這樣的網站,它有一個Fatal error在側邊欄或在屏幕中間的方框中嗎?你知道為什么會出現這種情況嗎?因為處理過程和輸出被混合在了一起。請看一些XML。
$xml = ""; $xml = ""; //Send xml data echo $xml; 0
工作正常。但它需要一些改進。
$xml = ""; $xml = ""; //Send xml data header("content-type: text/xml"); echo $xml; 0
請注意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:
". mysqli_connect_error()); } //Set the character set of the connection if(!mysqli_set_charset ( $c , "UTF8" )) { die("mysqli_set_charset() failed"); }一旦你連接到數據庫,不妨設置連接字符集。當你在你的應用程序中使用多種語言時,這絕對有必要。
否則會發生什么呢?你會在非英文文本中看到很多的方框和????????。
10.使用帶有正確字符集選項的htmlentitiesPHP 5.4之前,使用的默認字符編碼是ISO-8859-1,這不能顯示例如à a 這樣的字符。
$value = htmlentities($this->value , ENT_QUOTES , "UTF-8");從PHP 5.4起,默認編碼成了UTF-8,這解決了大部分的問題,但你最好還是知道這件事,如果你的應用程序使用多種語言的話。
11.不要在你的應用程序中gzip輸出,讓apache來做考慮使用ob_gzhandler?不,別這樣做。它沒有任何意義。PHP應該是來寫應用程序的。不要擔心PHP中有關如何優化在服務器和瀏覽器之間傳輸的數據。
使用apache mod_gzip/mod_deflate通過.htaccess文件壓縮內容。
12.從php echo javascript代碼時使用json_encode有些時候一些JavaScript代碼是從php動態生成的。
$images = array( "myself.png" , "friends.png" , "colleagues.png" ); $js_code = ""; foreach($images as $image) { $js_code .= ""$image" ,"; } $js_code = "var images = [" . $js_code . "]; "; echo $js_code; //Output is var images = ["myself.png" ,"friends.png" ,"colleagues.png" ,];放聰明點。使用json_encode:
$images = array( "myself.png" , "friends.png" , "colleagues.png" ); $js_code = "var images = " . json_encode($images); echo $js_code; //Output is : var images = ["myself.png","friends.png","colleagues.png"]這不是很整潔?
13.在寫入任何文件之前檢查目錄是否可寫在寫入或保存任何文件之前,請務必要檢查該目錄是否是可寫的,如果不可寫的話,會閃爍錯誤消息。這將節省你大量的“調試”時間。當你工作于Linux時,權限是必須要處理的,并且會有很多很多的權限問題時,當目錄不可寫,文件無法讀取等的時候。
請確保你的應用程序盡可能智能化,并在最短的時間內報告最重要的信息。
$contents = "All the content"; $file_path = "/var/www/project/content.txt"; file_put_contents($file_path , $contents);這完全正確。但有一些間接的問題。file_put_contents可能會因為一些原因而失敗:
父目錄不存在
目錄存在,但不可寫
鎖定文件用于寫入?
因此,在寫入文件之前最好能夠一切都弄明確。
$contents = "All the content"; $dir = "/var/www/project"; $file_path = $dir . "/content.txt"; if(is_writable($dir)) { file_put_contents($file_path , $contents); } else { die("Directory $dir is not writable, or does not exist. Please check"); }通過這樣做,你就能得到哪里文件寫入失敗以及為什么失敗的準確信息。
14.改變應用程序創建的文件的權限當在Linux環境下工作時,權限處理會浪費你很多時間。因此,只要你的php應用程序創建了一些文件,那就應該修改它們的權限以確保它們在外面“平易近人”。否則,例如,文件是由“php”用戶創建的,而你作為一個不同的用戶,系統就不會讓你訪問或打開文件,然后你必須努力獲得root權限,更改文件權限等等。
// Read and write for owner, read for everybody else chmod("/somedir/somefile", 0644); // Everything for owner, read and execute for others chmod("/somedir/somefile", 0755);15.不要檢查提交按鈕值來檢查表單提交if($_POST["submit"] == "Save") { //Save the things }以上代碼在大多數時候是正確的,除了應用程序使用多語言的情況。然后“Save”可以是很多不同的東西。那么你該如何再做比較?所以不能依靠提交按鈕的值。相反,使用這個:
if( $_SERVER["REQUEST_METHOD"] == "POST" and isset($_POST["submit"]) ) { //Save the things }現在你就可以擺脫提交按鈕的值了。
16.在函數中總是有相同值的地方使用靜態變量//Delay for some time function delay() { $sync_delay = get_option("sync_delay"); echo "
Delaying for $sync_delay seconds..."; sleep($sync_delay); echo "Done
"; }相反,使用靜態變量:
//Delay for some time function delay() { static $sync_delay = null; if($sync_delay == null) { $sync_delay = get_option("sync_delay"); } echo "17.不要直接使用$ _SESSION變量
Delaying for $sync_delay seconds..."; sleep($sync_delay); echo "Done
"; }一些簡單的例子是:
$_SESSION["username"] = $username; $username = $_SESSION["username"];但是這有一個問題。如果你正在相同域中運行多個應用程序,會話變量會發生沖突。2個不同的應用程序在會話變量中可能會設置相同的鍵名。舉個例子,一個相同域的前端門戶和后臺管理應用程序。
因此,用包裝函數使用應用程序特定鍵:
define("APP_ID" , "abc_corp_ecommerce"); //Function to get a session variable function session_get($key) { $k = APP_ID . "." . $key; if(isset($_SESSION[$k])) { return $_SESSION[$k]; } return false; } //Function set the session variable function session_set($key , $value) { $k = APP_ID . "." . $key; $_SESSION[$k] = $value; return true; }18.封裝實用輔助函數到一個類中所以,你必須在一個文件中有很多實用函數:
function utility_a() { //This function does a utility thing like string processing } function utility_b() { //This function does nother utility thing like database processing } function utility_c() { //This function is ... }自由地在應用程序中使用函數。那么你或許想要將它們包裝成一個類作為靜態函數:
class Utility { public static function utility_a() { } public static function utility_b() { } public static function utility_c() { } } //and call them as $a = Utility::utility_a(); $b = Utility::utility_b();這里你可以得到的一個明顯好處是,如果php有相似名稱的內置函數,那么名稱不會發生沖突。
從另一個角度看,你可以在相同的應用程序中保持多個版本的相同類,而不會發生任何沖突。因為它被封裝了,就是這樣。
19.一些傻瓜式技巧使用echo代替print
使用str_replace代替preg_replace,除非你確定需要它
不要使用short tags
對于簡單的字符串使用單引號代替雙引號
在header重定向之后要記得做一個exit
千萬不要把函數調用放到for循環控制行中。
isset比strlen快
正確和一致地格式化你的代碼
不要丟失循環或if-else塊的括號。
不要寫這樣的代碼:
if($a == true) $a_count++;這絕對是一種浪費。
這樣寫
if($a == true) { $a_count++; }不要通過吃掉語法縮短你的代碼。而是要讓你的邏輯更簡短。
使用具有代碼高亮功能的文本編輯器。代碼高亮有助于減少錯誤。
20. 使用array_map快速處理數組比方說,你要trim一個數組的所有元素。新手會這樣做:
foreach($arr as $c => $v) { $arr[$c] = trim($v); }但它可以使用array_map變得更整潔:
$arr = array_map("trim" , $arr);這適用于trim數組$arr的所有元素。另一個類似的函數是array_walk。
21.使用php過濾器驗證數據你是不是使用正則表達式來驗證如電子郵件,IP地址等值?是的,每個人都是這樣做的。現在,讓我們試試一個不同的東西,那就是過濾器。
php過濾器擴展程序將提供簡單的方法來有效驗證或校驗值。
22.強制類型檢查$amount = intval( $_GET["amount"] ); $rate = (int) $_GET["rate"];這是一種好習慣。
23.使用set_error_handler()將Php錯誤寫入到文件set_error_handler()可以用來設置自定義的錯誤處理程序。在文件中編寫一些重要的錯誤用于日志是個好主意。
24.小心處理大型數組大型的數組或字符串,如果一個變量保存了一些規模非常大的東西,那么要小心處理。常見錯誤是創建副本,然后耗盡內存,并得到內存溢出的致命錯誤:
$db_records_in_array_format; //This is a big array holding 1000 rows from a table each having 20 columns , every row is atleast 100 bytes , so total 1000 * 20 * 100 = 2MB $cc = $db_records_in_array_format; //2MB more some_function($cc); //Another 2MB ?當導入csv文件或導出表到csv文件時,上面這樣的代碼很常見。
像上面這樣做可能經常會由于內存限制而讓腳本崩潰。對于小規模的變量它不會出現問題,但當處理大型數組時一定要對此加以避免。
考慮通過引用傳遞它們,或者將它們存儲在一個類變量中:
$a = get_large_array(); pass_to_function(&$a);這樣一來,相同的變量(并非其副本)將用于該函數。
class A { function first() { $this->a = get_large_array(); $this->pass_to_function(); } function pass_to_function() { //process $this->a } }盡快復原它們,這樣內存就能被釋放,并且腳本的其余部分就能放松。
下面是關于如何通過引用來賦值從而節省內存的一個簡單示例。
"; $b = $a; $b[0] = "B"; echo "Memory usage in MB after 1st copy : ". memory_get_usage() / 1000000 . "
"; $c = $a; $c[0] = "B"; echo "Memory usage in MB after 2st copy : ". memory_get_usage() / 1000000 . "
"; $d =& $a; $d[0] = "B"; echo "Memory usage in MB after 3st copy (reference) : ". memory_get_usage() / 1000000 . "
";一個典型php 5.4機器上的輸出是:
Memory usage in MB : 18.08208 Memory usage in MB after 1st copy : 27.930944 Memory usage in MB after 2st copy : 37.779808 Memory usage in MB after 3st copy (reference) : 37.779864因此可以看出,內存被保存在第3份通過引用的副本中。否則,在所有普通副本中內存將被越來越多地使用。
25.在整個腳本中使用單一的數據庫連接請確保你在整個腳本使用單一的數據庫連接。從一開始就打開連接,使用至結束,并在結束時關閉它。不要像這樣在函數內打開連接:
function add_to_cart() { $db = new Database(); $db->query("INSERT INTO cart ....."); } function empty_cart() { $db = new Database(); $db->query("DELETE FROM cart ....."); }有多個連接也不好,會因為每個連接都需要時間來創建和使用更多的內存,而導致執行減緩。
在特殊情況下。例如數據庫連接,可以使用單例模式。
26.避免直接寫SQL,要把SQL語句抽象出來不厭其煩的寫了太多如下的語句:
$query = "INSERT INTO users(name , email , address , phone) VALUES("$name" , "$email" , "$address" , "$phone")"; $db->query($query); //call to mysqli_query()這不是個建壯的方案。它有些缺點:
每次都手動轉義值
驗證查詢是否正確
查詢的錯誤會花很長時間識別(除非每次都用if-else檢查)
很難維護復雜的查詢
因此使用函數封裝:
function insert_record($table_name , $data) { foreach($data as $key => $value) { //mysqli_real_escape_string $data[$key] = $db->mres($value); } $fields = implode("," , array_keys($data)); $values = """ . implode("","" , array_values($data)) . """; //Final query $query = "INSERT INTO {$table}($fields) VALUES($values)"; return $db->query($query); } $data = array("name" => $name , "email" => $email , "address" => $address , "phone" => $phone); insert_record("users" , $data);看到了嗎?這樣會更易讀和擴展。record_data函數小心的處理了轉義。
27.將數據庫生成的內容緩存到靜態文件中
最大的優點是數據被預處理為一個數組,任何語法錯誤都會被捕獲。
該函數應該定義在某個database類中,你可以像$db->insert_record這樣調用。
類似的也可以編寫update,select,delete方法。試試吧。如果所有的內容都是從數據庫獲取的,它們應該被緩存。一旦生成了,就將它們保存在臨時文件中。下次請求該頁面時,可直接從緩存中取,不用再查數據庫。
好處:
節約php處理頁面的時間, 執行更快
更少的數據庫查詢意味著更少的mysql連接開銷
28.在數據庫中保存session基于文件的session策略會有很多限制。使用基于文件的session不能擴展到集群中,因為session保存在單個服務器中。但數據庫可被多個服務器訪問。這樣就可以解決問題。
在數據庫中保存session數據,還有更多好處:處理username重復登錄問題。同個username不能在兩個地方同時登錄。
能更準備的查詢在線用戶狀態。
29. 避免使用全局變量使用defines/constants
使用函數獲取值
使用類并通過$this訪問
30.在head中使用base標簽沒聽說過?請看下面: