40+個對初學者非常有用的PHP技巧(一)


摘要:今天我們要介紹一些關於改善和優化PHP代碼的提示和技巧。請注意,這些PHP技巧適用於初學者,而不是那些已經在使用MVC框架的人。

今天我們要介紹一些關於改善和優化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.通過會話傳播狀態消息

狀態消息是那些執行任務後生成的消息。

<?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,這解決了大部分的問題,但你最好還是知道這件事,如果你的應用程序使用多種語言的話。

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 "<br />Delaying for $sync_delay seconds...";
    sleep($sync_delay);
    echo "Done <br />";
}

相反,使用靜態變量:

//Delay for some time
function delay()
{
    static $sync_delay = null;

    if($sync_delay == null)
    {
    $sync_delay = get_option('sync_delay');
    }

    echo "<br />Delaying for $sync_delay seconds...";
    sleep($sync_delay);
    echo "Done <br />";
}

17.不要直接使用$ _SESSION變量

一些簡單的例子是:

$_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
    }
}

儘快復原它們,這樣內存就能被釋放,並且腳本的其餘部分就能放鬆。

下面是關於如何通過引用來賦值從而節省內存的一個簡單示例。

<?php

ini_set('display_errors' , true);
error_reporting(E_ALL);

$a = array();

for($i = 0; $i < 100000 ; $i++)
{
    $a[$i] = 'A'.$i;
}

echo 'Memory usage in MB : '. memory_get_usage() / 1000000 . '<br />';

$b = $a;
$b[0] = 'B';

echo 'Memory usage in MB after 1st copy : '. memory_get_usage() / 1000000 . '<br />';

$c = $a;
$c[0] = 'B';

echo 'Memory usage in MB after 2st copy : '. memory_get_usage() / 1000000 . '<br />';

$d =& $a;
$d[0] = 'B';

echo 'Memory usage in MB after 3st copy (reference) : '. memory_get_usage() / 1000000 . '<br />';

一個典型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 .....");
}

有多個連接也不好,會因爲每個連接都需要時間來創建和使用更多的內存,而導致執行減緩。

在特殊情況下。例如數據庫連接,可以使用單例模式。



發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章