截至目前(2014.2), PHP 的最新穩定版本是 PHP5.5, 但有差不多一半的用戶仍在使用已經不在維護[注]的 PHP5.2, 其餘的一半用戶在使用 PHP5.3[注]
因爲 PHP 那“集百家之長”的蛋疼語法,加上社區氛圍不好,很多人對新版本,新特徵並無興趣。
本文將會介紹自 PHP5.2 起,直至 PHP5.6 中增加的新特徵。
PHP5.2 以前:autoload, PDO 和 MySQLi, 類型約束
PHP5.2:JSON 支持
PHP5.3:棄用的功能,匿名函數,新增魔術方法,命名空間,後期靜態綁定,Heredoc 和 Nowdoc, const, 三元運算符,Phar
PHP5.4:Short Open Tag, 數組簡寫形式,Traits, 內置 Web 服務器,細節修改
PHP5.5:yield, list() 用於 foreach, 細節修改
PHP5.6: 常量增強,可變函數參數,命名空間增強
注:已於2011年1月停止支持: http://www.php.net/eol.php
注:http://w3techs.com/technologies/details/pl-php/5/all
PHP5.2以前
(2006前)
順便介紹一下 PHP5.2 已經出現但值得介紹的特徵。
autoload
大家可能都知道 __autoload() 函數,如果定義了該函數,那麼當在代碼中使用一個未定義的類的時候,該函數就會被調用,你可以在該函數中加載相應的類實現文件,如:
1
2
3
|
function
__autoload( $classname ) { require_once ( "{$classname}.php" ) } |
但該函數已經不被建議使用,原因是一個項目中僅能有一個這樣的 __autoload() 函數,因爲 PHP 不允許函數重名。但當你使用一些類庫的時候,難免會出現多個 autoload 函數的需要,於是 spl_autoload_register() 取而代之:
1
2
3
4
|
spl_autoload_register( function ( $classname ) { require_once ( "{$classname}.php" ) }); |
spl_autoload_register() 會將一個函數註冊到 autoload 函數列表中,當出現未定義的類的時候,SPL [注] 會按照註冊的倒序逐個調用被註冊的 autoload 函數,這意味着你可以使用 spl_autoload_register() 註冊多個 autoload 函數.
注:SPL: Standard PHP Library, 標準 PHP 庫, 被設計用來解決一些經典問題(如數據結構).
PDO 和 MySQLi
即 PHP Data Object, PHP 數據對象,這是 PHP 的新式數據庫訪問接口。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
按照傳統的風格,訪問 MySQL 數據庫應該是這樣子: //連接到服務器,選擇數據庫 $conn
= mysql_connect( "localhost" ,
"user" ,
"password" ); mysql_select_db( "database" ); // 執行 SQL 查詢 $type
= $_POST [ 'type' ]; $sql
= "SELECT * FROM `table` WHERE `type` = {$type}" ; $result
= mysql_query( $sql ); // 打印結果 while ( $row
= mysql_fetch_array( $result , MYSQL_ASSOC)) { foreach ( $row
as
$k => $v ) print
"{$k}: {$v}\n" ; } // 釋放結果集,關閉連接 mysql_free_result( $result ); mysql_close( $conn ); |
爲了能夠讓代碼實現數據庫無關,即一段代碼同時適用於多種數據庫(例如以上代碼僅僅適用於MySQL),PHP 官方設計了 PDO.
除此之外,PDO 還提供了更多功能,比如:
• 面向對象風格的接口
• SQL預編譯(prepare), 佔位符語法
• 更高的執行效率,作爲官方推薦,有特別的性能優化
• 支持大部分SQL數據庫,更換數據庫無需改動代碼
上面的代碼用 PDO 實現將會是這樣:
1
2
3
4
5
6
7
8
9
10
11
|
// 連接到數據庫 $conn
= new
PDO( "mysql:host=localhost;dbname=database" ,
"user" ,
"password" ); // 預編譯SQL, 綁定參數 $query
= $conn ->prepare( "SELECT * FROM `table` WHERE `type` = :type" ); $query ->bindParam( "type" ,
$_POST [ 'type' ]); // 執行查詢並打印結果 foreach ( $query ->execute()
as
$row ) { foreach ( $row
as
$k => $v ) print
"{$k}: {$v}\n" ; } |
PDO 是官方推薦的,更爲通用的數據庫訪問方式,如果你沒有特殊需求,那麼你最好學習和使用 PDO.
但如果你需要使用 MySQL 所特有的高級功能,那麼你可能需要嘗試一下 MySQLi, 因爲 PDO 爲了能夠同時在多種數據庫上使用,不會包含那些 MySQL 獨有的功能。
MySQLi 是 MySQL 的增強接口,同時提供面向過程和麪向對象接口,也是目前推薦的 MySQL 驅動,舊的C風格 MySQL 接口將會在今後被默認關閉。
MySQLi 的用法和以上兩段代碼相比,沒有太多新概念,在此不再給出示例,可以參見 PHP 官網文檔 [注]。
注:http://www.php.net/manual/en/mysqli.quickstart.php
類型約束
通過類型約束可以限制參數的類型,不過這一機制並不完善,目前僅適用於類和 callable(可執行類型) 以及 array(數組), 不適用於 string 和 int.
1
2
3
4
5
|
// 限制第一個參數爲 MyClass, 第二個參數爲可執行類型,第三個參數爲數組 function
MyFunction(MyClass $a , callable
$b ,
array
$c ) { // ... } |
PHP5.2
(2006-2011)
JSON 支持
包括 json_encode(), json_decode() 等函數,JSON 算是在 Web 領域非常常用的數據交換格式,可以被 JS 直接支持,JSON 實際上是 JS 語法的一部分。
JSON 系列函數,可以將 PHP 中的數組結構與 JSON 字符串進行轉換:
1
2
3
4
5
|
$array
= array ( "key"
=> "value" ,
"array"
=> array (1, 2, 3, 4)); $json
= json_encode( $array ); echo
"{$json}\n" ; $object
= json_decode( $json ); print_r( $object ); |
輸出:
1
2
3
4
5
6
7
8
9
10
11
12
|
{"key":"value","array":[1,2,3,4]} stdClass Object ( [key] => value [array] => Array ( [0] => 1 [1] => 2 [2] => 3 [3] => 4 ) ) |
值得注意的是 json_decode() 默認會返回一個對象而非數組,如果需要返回數組需要將第二個參數設置爲 true.
PHP5.3
(2009-2012)
PHP5.3 算是一個非常大的更新,新增了大量新特徵,同時也做了一些不向下兼容的修改。
棄用的功能
以下幾個功能被棄用,若在配置文件中啓用,則 PHP 會在運行時發出警告。
Register Globals
這是 php.ini 中的一個選項(register_globals), 開啓後會將所有表單變量($_GET和$_POST)註冊爲全局變量.
看下面的例子:
1
2
3
4
|
if (isAuth()) $authorized
= true; if ( $authorized ) include ( "page.php" ); |
這段代碼在通過驗證時,將 $authorized 設置爲 true. 然後根據 $authorized 的值來決定是否顯示頁面.
但由於並沒有事先把 $authorized 初始化爲 false, 當 register_globals 打開時,可能訪問 /auth.php?authorized=1 來定義該變量值,繞過身份驗證。
該特徵屬於歷史遺留問題,在 PHP4.2 中被默認關閉,在 PHP5.4 中被移除。
Magic Quotes
對應 php.ini 中的選項 magic_quotes_gpc, 這個特徵同樣屬於歷史遺留問題,已經在 PHP5.4 中移除。
該特徵會將所有用戶輸入進行轉義,這看上去不錯,在第一章我們提到過要對用戶輸入進行轉義。
但是 PHP 並不知道哪些輸入會進入 SQL , 哪些輸入會進入 Shell, 哪些輸入會被顯示爲 HTML, 所以很多時候這種轉義會引起混亂。
Safe Mode
很多虛擬主機提供商使用 Safe Mode 來隔離多個用戶,但 Safe Mode 存在諸多問題,例如某些擴展並不按照 Safe Mode 來進行權限控制。
PHP官方推薦使用操作系統的機制來進行權限隔離,讓Web服務器以不同的用戶權限來運行PHP解釋器,請參見第一章中的最小權限原則.
匿名函數
也叫閉包(Closures), 經常被用來臨時性地創建一個無名函數,用於回調函數等用途。
1
2
3
4
5
|
$func
= function ( $arg ) { print
$arg ; }; $func ( "Hello World" ); |
以上代碼定義了一個匿名函數,並賦值給了 $func.
可以看到定義匿名函數依舊使用 function 關鍵字,只不過省略了函數名,直接是參數列表。
然後我們又調用了 $func 所儲存的匿名函數。
匿名函數還可以用 use 關鍵字來捕捉外部變量:
1
2
3
4
5
6
|
function
arrayPlus( $array ,
$num ) { array_walk ( $array ,
function (& $v )
use ( $num ){ $v
+= $num ; }); } |
上面的代碼定義了一個 arrayPlus() 函數(這不是匿名函數), 它會將一個數組($array)中的每一項,加上一個指定的數字($num).
在 arrayPlus() 的實現中,我們使用了 array_walk() 函數,它會爲一個數組的每一項執行一個回調函數,即我們定義的匿名函數。
在匿名函數的參數列表後,我們用 use 關鍵字將匿名函數外的 $num 捕捉到了函數內,以便知道到底應該加上多少。
魔術方法:__invoke(), __callStatic()
PHP 的面向對象體系中,提供了若干“魔術方法”,用於實現類似其他語言中的“重載”,如在訪問不存在的屬性、方法時觸發某個魔術方法。
隨着匿名函數的加入,PHP 引入了一個新的魔術方法 __invoke().
該魔術方法會在將一個對象作爲函數調用時被調用:
1
2
3
4
5
6
7
8
9
|
class
A { public
function
__invoke( $str ) { print
"A::__invoke(): {$str}" ; } } $a
= new
A; $a ( "Hello World" ); |
輸出毫無疑問是:
1
|
A::__invoke(): Hello World |
__callStatic() 則會在調用一個不存在的靜態方法時被調用。
命名空間
PHP的命名空間有着前無古人後無來者的無比蛋疼的語法:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
// 命名空間的分隔符是反斜槓,該聲明語句必須在文件第一行。 // 命名空間中可以包含任意代碼,但只有 **類, 函數, 常量** 受命名空間影響。 namespace
XXOO\Test; // 該類的完整限定名是 \XXOO\Test\A , 其中第一個反斜槓表示全局命名空間。 class
A{} // 你還可以在已經文件中定義第二個命名空間,接下來的代碼將都位於 \Other\Test2 . namespace
Other\Test2; // 實例化來自其他命名空間的對象: $a
= new
\XXOO\Test\A; class
B{} // 你還可以用花括號定義第三個命名空間 namespace
Other { // 實例化來自子命名空間的對象: $b
= new
Test2\B; // 導入來自其他命名空間的名稱,並重命名, // 注意只能導入類,不能用於函數和常量。 use
\XXOO\Test\A as
ClassA } |
更多有關命名空間的語法介紹請參見官網 [注].
命名空間時常和 autoload 一同使用,用於自動加載類實現文件:
1
2
3
4
5
|
spl_autoload_register( function
( $class ) { spl_autoload( str_replace ( "\\" ,
"/" ,
$class )); } ); |
當你實例化一個類 \XXOO\Test\A 的時候,這個類的完整限定名會被傳遞給 autoload 函數,autoload 函數將類名中的命名空間分隔符(反斜槓)替換爲斜槓,幷包含對應文件。
這樣可以實現類定義文件分級儲存,按需自動加載。
注:http://www.php.net/manual/zh/language.namespaces.php
後期靜態綁定
PHP 的 OPP 機制,具有繼承和類似虛函數的功能,例如如下的代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
class
A { public
function
callFuncXXOO() { print
$this ->funcXXOO(); } public
function
funcXXOO() { return
"A::funcXXOO()" ; } } class
B extends
A { public
function
funcXXOO() { return
"B::funcXXOO" ; } } $b
= new
B; $b ->callFuncXXOO(); |
輸出是:
1
|
B::funcXXOO |
可以看到,當在 A 中使用 $this->funcXXOO() 時,體現了“虛函數”的機制,實際調用的是 B::funcXXOO().
然而如果將所有函數都改爲靜態函數:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
class
A { static
public
function callFuncXXOO() { print
self::funcXXOO(); } static
public
function funcXXOO() { return
"A::funcXXOO()" ; } } class
B extends
A { static
public
function funcXXOO() { return
"B::funcXXOO" ; } } $b
= new
B; $b ->callFuncXXOO(); |
情況就沒這麼樂觀了,輸出是:
1
|
A::funcXXOO() |
這是因爲 self 的語義本來就是“當前類”,所以 PHP5.3 給 static 關鍵字賦予了一個新功能:後期靜態綁定:
1
2
3
4
5
6
7
8
9
|
class
A { static
public
function callFuncXXOO() { print
static ::funcXXOO(); } // ... } // ... |
這樣就會像預期一樣輸出了:
1
|
B::funcXXOO |
Heredoc 和 Nowdoc
PHP5.3 對 Heredoc 以及 Nowdoc 進行了一些改進,它們都用於在 PHP 代碼中嵌入大段字符串。
Heredoc 的行爲類似於一個雙引號字符串:
1
2
3
4
|
$name
= "MyName" ; echo
<<< TEXT My name is
"{$name}" . TEXT; |
Heredoc 以三個左尖括號開始,後面跟一個標識符(TEXT), 直到一個同樣的頂格的標識符(不能縮進)結束。
就像雙引號字符串一樣,其中可以嵌入變量。
Heredoc 還可以用於函數參數,以及類成員初始化:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
var_dump(<<<EOD Hello World EOD ); class
A { const
xx = <<< EOD Hello World EOD; public
$oo = <<< EOD Hello World EOD; } |
Nowdoc 的行爲像一個單引號字符串,不能在其中嵌入變量,和 Heredoc 唯一的區別就是,三個左尖括號後的標識符要以單引號括起來:
1
2
3
4
|
$name
= "MyName" ; echo
<<< 'TEXT' My name is
"{$name}" . TEXT; |
輸出:
1
|
My name is
"{$name}" . |
用 const 定義常量
PHP5.3 起同時支持在全局命名空間和類中使用 const 定義常量。
舊式風格:
1
|
define( "XOOO" ,
"Value" ); |
新式風格:
1
|
const
XXOO = "Value" ; |
const 形式僅適用於常量,不適用於運行時才能求值的表達式:
1
2
3
4
|
// 正確 const
XXOO = 1234; // 錯誤 const
XXOO = 2 * 617; |
三元運算符簡寫形式
舊式風格:
1
|
echo
$a ?
$a :
"No Value" ; |
可簡寫成:
1
|
echo
$a ?:
"No Value" ; |
即如果省略三元運算符的第二個部分,會默認用第一個部分代替。
Phar
Phar即PHP Archive, 起初只是Pear中的一個庫而已,後來在PHP5.3被重新編寫成C擴展並內置到 PHP 中。
Phar用來將多個 .php 腳本打包(也可以打包其他文件)成一個 .phar 的壓縮文件(通常是ZIP格式)。
目的在於模仿 Java 的 .jar, 不對,目的是爲了讓發佈PHP應用程序更加方便。同時還提供了數字簽名驗證等功能。
.phar 文件可以像 .php 文件一樣,被PHP引擎解釋執行,同時你還可以寫出這樣的代碼來包含(require) .phar 中的代碼:
1
2
|
require ( "xxoo.phar" ); |
更多信息請參見官網 [注].
注:http://www.php.net/manual/zh/phar.using.intro.php
PHP5.4
(2012-2013)
Short Open Tag
Short Open Tag 自 PHP5.4 起總是可用。
在這裏集中講一下有關 PHP 起止標籤的問題。即:
1
2
3
|
<?php // Code... ?> |
通常就是上面的形式,除此之外還有一種簡寫形式:
1
|
<?
/* Code... */
?> |
還可以把
1
|
<?php
echo
$xxoo ;?> |
簡寫成:
1
|
<?=
$xxoo ;?> |
這種簡寫形式被稱爲 Short Open Tag, 在 PHP5.3 起被默認開啓,在 PHP5.4 起總是可用。
使用這種簡寫形式在 HTML 中嵌入 PHP 變量將會非常方便。
對於純 PHP 文件(如類實現文件), PHP 官方建議頂格寫起始標記,同時 省略 結束標記。
這樣可以確保整個 PHP 文件都是 PHP 代碼,沒有任何輸出,否則當你包含該文件後,設置 Header 和 Cookie 時會遇到一些麻煩 [注].
注:Header 和 Cookie 必須在輸出任何內容之前被髮送。
數組簡寫形式
這是非常方便的一項特徵!
1
2
3
4
|
// 原來的數組寫法 $arr
= array ( "key"
=> "value" ,
"key2"
=> "value2" ); // 簡寫形式 $arr
= [ "key"
=> "value" ,
"key2"
=> "value2" ]; |
Traits
所謂Traits就是“構件”,是用來替代繼承的一種機制。PHP中無法進行多重繼承,但一個類可以包含多個Traits.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
// Traits不能被單獨實例化,只能被類所包含 trait SayWorld { public
function
sayHello() { echo
'World!' ; } } class
MyHelloWorld { // 將SayWorld中的成員包含進來 use
SayWorld; } $xxoo
= new
MyHelloWorld(); // sayHello() 函數是來自 SayWorld 構件的 $xxoo ->sayHello(); |
Traits還有很多神奇的功能,比如包含多個Traits, 解決衝突,修改訪問權限,爲函數設置別名等等。
Traits中也同樣可以包含Traits. 篇幅有限不能逐個舉例,詳情參見官網 [注].
注:http://www.php.net/manual/zh/language.oop5.traits.php
內置 Web 服務器
PHP從5.4開始內置一個輕量級的Web服務器,不支持併發,定位是用於開發和調試環境。
在開發環境使用它的確非常方便。
1
|
php -S localhost:8000 |
這樣就在當前目錄建立起了一個Web服務器,你可以通過 http://localhost:8000/ 來訪問。
其中localhost是監聽的ip,8000是監聽的端口,可以自行修改。
很多應用中,都會進行URL重寫,所以PHP提供了一個設置路由腳本的功能:
1
|
php -S localhost:8000 index.php |
這樣一來,所有的請求都會由index.php來處理。
你還可以使用 XDebug 來進行斷點調試。
細節修改
PHP5.4 新增了動態訪問靜態方法的方式:
1
2
|
$func
= "funcXXOO" ; A::{ $func }(); |
新增在實例化時訪問類成員的特徵:
1
|
( new
MyClass)->xxoo(); |
新增支持對函數返回數組的成員訪問解析(這種寫法在之前版本是會報錯的):
1
|
print
func()[0]; |
PHP5.5
(2013起)
yield
yield關鍵字用於當函數需要返回一個迭代器的時候, 逐個返回值。
1
2
3
4
5
|
function
number10() { for ( $i
= 1; $i
<= 10; $i
+= 1) yield
$i ; } |
該函數的返回值是一個數組:
list() 用於 foreach
可以用 list() 在 foreach 中解析嵌套的數組:
1
2
3
4
5
6
|
$array = [ [1, 2, 3], [4, 5, 6], ]; foreach ($array as list($a, $b, $c)) echo "{$a} {$b} {$c}\n"; |
結果:
1
2
|
1 2 3 4 5 6 |
細節修改
不推薦使用 mysql 函數,推薦使用 PDO 或 MySQLi, 參見前文。
不再支持Windows XP.
可用 MyClass::class 取到一個類的完整限定名(包括命名空間)。
empty() 支持表達式作爲參數。
try-catch 結構新增 finally 塊。
PHP5.6
更好的常量
定義常量時允許使用之前定義的常量進行計算:
1
2
3
4
5
6
7
|
const
A = 2; const
B = A + 1; class
C { const
STR = "hello" ; const
STR2 = self::STR + ", world" ; } |
允許常量作爲函數參數默認值:
1
|
function
func( $arg
= C::STR2) |
更好的可變函數參數
用於代替 func_get_args()
1
2
3
4
5
6
7
|
function
add(... $args ) { $result
= 0; foreach ( $args
as
$arg ) $result
+= $arg ; return
$result ; } |
同時可以在調用函數時,把數組展開爲函數參數:
1
2
3
|
$arr
= [2, 3]; add(1, ... $arr ); // 結果爲 6 |
命名空間
命名空間支持常量和函數:
1
2
3
4
5
6
7
8
9
10
|
namespace
Name\Space { const
FOO = 42; function
f() { echo
__FUNCTION__ . "\n" ; } } namespace
{ use
const
Name\Space\FOO; use
function
Name\Space\f; echo
FOO. "\n" ; f(); } |
本篇文章來源於 Linux公社網站(www.linuxidc.com)
原文鏈接:http://www.linuxidc.com/Linux/2014-02/96257.htm