PHP5各個版本的新功能和新特性總結_PHP教程

來源:互聯網
上載者:User
本文目錄:
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: 常量增強,可變函數參數,命名空間增強

一、PHP5.2以前(2006前)
順便介紹一下 PHP5.2 已經出現但值得介紹的特徵。
autoload
大家可能都知道 __autoload() 函數,如果定義了該函數,那麼當在代碼中使用一個未定義的類的時候,該函數就會被調用,你可以在該函數中載入相應的類實現檔案,如:
複製代碼 代碼如下:function __autoload($classname)
{
require_once("{$classname}.php")
}
但該函數已經不被建議使用,原因是一個項目中僅能有一個這樣的 __autoload() 函數,因為 PHP 不允許函數重名。但當你使用一些類庫的時候,難免會出現多個 autoload 函數的需要,於是 spl_autoload_register() 取而代之:
複製代碼 代碼如下: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 的新式資料庫提供者。
按照傳統的風格,訪問 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 還提供了更多功能,比如:

1.物件導向風格的介面
2.SQL先行編譯(prepare), 預留位置文法
3.更高的執行效率,作為官方推薦,有特別的效能最佳化
4.支援大部分SQL資料庫,更換資料庫無需改動代碼

上面的代碼用 PDO 實現將會是這樣:
複製代碼 代碼如下:// 串連到資料庫
$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.
複製代碼 代碼如下:// 限制第一個參數為 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 字串進行轉換:
複製代碼 代碼如下:$array = ["key" => "value", "array" => [1, 2, 3, 4]];
$json = json_encode($array);
echo "{$json}\n";

$object = json_decode($json);
print_r($object);
輸出:
複製代碼 代碼如下:{"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 算是一個非常大的更新,新增了大量新特徵,同時也做了一些不向下相容的修改。
【PHP5.3棄用的功能】:以下幾個功能被棄用,若在設定檔中啟用,則 PHP 會在運行時發出警告。

Register Globals
這是 php.ini 中的一個選項(register_globals), 開啟後會將所有表單變數($_GET和$_POST)註冊為全域變數.
看下面的例子:
複製代碼 代碼如下: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解譯器,請參見第一章中的最小許可權原則.

【PHP5.3的新增、改進】

匿名函數
也叫閉包(Closures), 經常被用來臨時性地建立一個無名函數,用於回呼函數等用途。
複製代碼 代碼如下:$func = function($arg)
{
print $arg;
};

$func("Hello World");
以上代碼定義了一個匿名函數,並賦值給了 $func.
可以看到定義匿名函數依舊使用 function 關鍵字,只不過省略了函數名,直接是參數列表。
然後我們又調用了 $func 所儲存的匿名函數。
匿名函數還可以用 use 關鍵字來捕捉外部變數:
複製代碼 代碼如下: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().
該魔術方法會在將一個對象作為函數調用時被調用:
複製代碼 代碼如下:class A
{
public function __invoke($str)
{
print "A::__invoke(): {$str}";
}
}

$a = new A;
$a("Hello World");
輸出毫無疑問是:
複製代碼 代碼如下:A::__invoke(): Hello World
__callStatic() 則會在調用一個不存在的靜態方法時被調用。


命名空間
PHP的命名空間有著前無古人後無來者的無比蛋疼的文法:
複製代碼 代碼如下:// 命名空間的分隔字元是反斜線,該聲明語句必須在檔案第一行。
// 命名空間中可以包含任意代碼,但只有 **類, 函數, 常量** 受命名空間影響。
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 一同使用,用於自動載入類實現檔案:


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 機制,具有繼承和類似虛函數的功能,例如如下的代碼:
複製代碼 代碼如下: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();
輸出是:
複製代碼 代碼如下:B::funcXXOO
可以看到,當在 A 中使用 $this->funcXXOO() 時,體現了“虛函數”的機制,實際調用的是 B::funcXXOO().
然而如果將所有函數都改為靜態函數:
複製代碼 代碼如下: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();
情況就沒這麼樂觀了,輸出是:
複製代碼 代碼如下:A::funcXXOO()
這是因為 self 的語義本來就是“當前類”,所以 PHP5.3 給 static 關鍵字賦予了一個新功能:後期靜態繫結:
複製代碼 代碼如下:class A
{
static public function callFuncXXOO()
{
print static::funcXXOO();
}

// ...
}

// ...
這樣就會像預期一樣輸出了:
複製代碼 代碼如下:B::funcXXOO

Heredoc 和 Nowdoc

PHP5.3 對 Heredoc 以及 Nowdoc 進行了一些改進,它們都用於在 PHP 代碼中嵌入大段字串。
Heredoc 的行為類似於一個雙引號字串:
複製代碼 代碼如下:$name = "MyName";
echo <<< TEXT
My name is "{$name}".
TEXT;
Heredoc 以三個左角括弧開始,後面跟一個標識符(TEXT), 直到一個同樣的頂格的標識符(不能縮排)結束。
就像雙引號字串一樣,其中可以嵌入變數。

Heredoc 還可以用於函數參數,以及類成員初始化:
複製代碼 代碼如下:var_dump(<<Hello World
EOD
);

class A
{
const xx = <<< EOD
Hello World
EOD;

public $oo = <<< EOD
Hello World
EOD;
}
Nowdoc 的行為像一個單引號字串,不能在其中嵌入變數,和 Heredoc 唯一的區別就是,三個左角括弧後的標識符要以單引號括起來:
複製代碼 代碼如下:$name = "MyName";
echo <<< 'TEXT'
My name is "{$name}".
TEXT;
輸出:
複製代碼 代碼如下:My name is "{$name}".

用 const 定義常量

PHP5.3 起同時支援在全域命名空間和類中使用 const 定義常量。
舊式風格:
複製代碼 代碼如下:define("XOOO", "Value");
新式風格:
const XXOO = "Value";
const 形式僅適用於常量,不適用於運行時才能求值的運算式:
複製代碼 代碼如下:// 正確
const XXOO = 1234;
// 錯誤
const XXOO = 2 * 617;

三元運算子簡寫形式
舊式風格:
複製代碼 代碼如下:echo $a ? $a : "No Value";
可簡寫成:
複製代碼 代碼如下: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 中的代碼:
複製代碼 代碼如下:require("xxoo.phar");
require("phar://xxoo.phar/xo/ox.php");
更多資訊請參見官網 [注].
註:http://www.php.net/manual/zh/phar.using.intro.php

PHP5.4(2012-2013)

Short Open Tag
Short Open Tag 自 PHP5.4 起總是可用。
在這裡集中講一下有關 PHP 起止標籤的問題。即:
複製代碼 代碼如下:// Code...
?>
通常就是上面的形式,除此之外還有一種簡寫形式:
複製代碼 代碼如下:
還可以把
複製代碼 代碼如下:
簡寫成:
複製代碼 代碼如下:
這種簡寫形式被稱為 Short Open Tag, 在 PHP5.3 起被預設開啟,在 PHP5.4 起總是可用。
使用這種簡寫形式在 HTML 中嵌入 PHP 變數將會非常方便。

對於純 PHP 檔案(如類實現檔案), PHP 官方建議頂格寫起始標記,同時 省略 結束標記。
這樣可以確保整個 PHP 檔案都是 PHP 代碼,沒有任何輸出,否則當你包含該檔案後,設定 Header 和 Cookie 時會遇到一些麻煩 [注].

註:Header 和 Cookie 必須在輸出任何內容之前被發送。

數組簡寫形式
這是非常方便的一項特徵!
複製代碼 代碼如下:// 原來的數組寫法
$arr = array("key" => "value", "key2" => "value2");
// 簡寫形式
$arr = ["key" => "value", "key2" => "value2"];

Traits
所謂Traits就是“構件”,是用來替代繼承的一種機制。PHP中無法進行多重繼承,但一個類可以包含多個Traits.
複製代碼 代碼如下:// 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伺服器,不支援並發,定位是用於開發和調試環境。
在開發環境使用它的確非常方便。
複製代碼 代碼如下:php -S localhost:8000
這樣就在目前的目錄建立起了一個Web伺服器,你可以通過 http://localhost:8000/ 來訪問。
其中localhost是監聽的ip,8000是監聽的連接埠,可以自行修改。

很多應用中,都會進行URL重寫,所以PHP提供了一個設定路由指令碼的功能:
複製代碼 代碼如下:php -S localhost:8000 index.php
這樣一來,所有的請求都會由index.php來處理。
你還可以使用 XDebug 來進行斷點調試。

細節修改
PHP5.4 新增了動態訪問靜態方法的方式:
複製代碼 代碼如下:$func = "funcXXOO";
A::{$func}();
新增在執行個體化時訪問類成員的特徵:
複製代碼 代碼如下:(new MyClass)->xxoo();
新增支援對函數返回數組的成員訪問解析(這種寫法在之前版本是會報錯的):
複製代碼 代碼如下:print func()[0];

PHP5.5(2013起)

yield
yield關鍵字用於當函數需要返回一個迭代器的時候, 逐個傳回值。
複製代碼 代碼如下:function number10()
{
for($i = 1; $i <= 10; $i += 1)
yield $i;
}
該函數的傳回值是一個數組:
複製代碼 代碼如下:[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
list() 用於 foreach
可以用 list() 在 foreach 中解析嵌套的數組:
複製代碼 代碼如下:$array = [
[1, 2, 3],
[4, 5, 6],
];

foreach ($array as list($a, $b, $c))
echo "{$a} {$b} {$c}\n";
結果:
複製代碼 代碼如下:1 2 3
4 5 6
細節修改
不推薦使用 mysql 函數,推薦使用 PDO 或 MySQLi, 參見前文。
不再支援Windows XP.
可用 MyClass::class 取到一個類的完整限定名(包括命名空間)。
empty() 支援運算式作為參數。
try-catch 結構新增 finally 塊。

PHP5.6

更好的常量
定義常量時允許使用之前定義的常量進行計算:
複製代碼 代碼如下:const A = 2;
const B = A + 1;

class C
{
const STR = "hello";
const STR2 = self::STR + ", world";
}
允許常量作為函數參數預設值:
複製代碼 代碼如下:function func($arg = C::STR2)

更好的可變函數參數
用於代替 func_get_args()
複製代碼 代碼如下:function add(...$args)
{
$result = 0;
foreach($args as $arg)
$result += $arg;
return $result;
}
同時可以在調用函數時,把數組展開為函數參數:
複製代碼 代碼如下:$arr = [2, 3];
add(1, ...$arr);
// 結果為 6
命名空間
命名空間支援常量和函數:
複製代碼 代碼如下: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();
}

http://www.bkjia.com/PHPjc/742139.htmlwww.bkjia.comtruehttp://www.bkjia.com/PHPjc/742139.htmlTechArticle本文目錄: PHP5.2 以前:autoload, PDO 和 MySQLi, 類型約束 PHP5.2:JSON 支援 PHP5.3:棄用的功能,匿名函數,新增魔術方法,命名空間,後期靜態...

  • 聯繫我們

    該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

    如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

    A Free Trial That Lets You Build Big!

    Start building with 50+ products and up to 12 months usage for Elastic Compute Service

    • Sales Support

      1 on 1 presale consultation

    • After-Sales Support

      24/7 Technical Support 6 Free Tickets per Quarter Faster Response

    • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.