日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > php >内容正文

php

PHP5.2至5.6的新增功能详解

發布時間:2024/9/19 php 28 豆豆
生活随笔 收集整理的這篇文章主要介紹了 PHP5.2至5.6的新增功能详解 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

概述:

php5.3不但引進了匿名函數還有更多更好多新的特性了,下面我們一起來了解一下PHP匿名函數與注意事項,具體內容如下

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:常量增強,可變函數參數,命名空間增強

?

截至目前(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() 函數,如果定義了該函數,那么當在代碼中使用一個未定義的類的時候,該函數就會被調用,你可以在該函數中加載相應的類實現文件,如:

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 還提供了更多功能,比如:

  • 面向對象風格的接口
  • SQL預編譯(prepare), 占位符語法
  • 更高的執行效率,作為官方推薦,有特別的性能優化
  • 支持大部分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 = array("key" => "value", "array" => 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 算是一個非常大的更新,新增了大量新特征,同時也做了一些不向下兼容的修改。

棄用的功能

以下幾個功能被棄用,若在配置文件中啟用,則 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解釋器,請參見第一章中的最小權限原則.

匿名函數

也叫閉包(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 捕捉到了函數內,以便知道到底應該加上多少。

允許 臨時創建一個沒有指定名稱的函數。最經常用作回調函數(callback)參數的值。當然,也有其它應用的情況。

<?php echo preg_replace_callback('~-([a-z])~', function ($match) { return strtoupper($match[1]); }, 'hello-world'); // 輸出 helloWorld ?>

魔術方法:__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(<<<EODHello 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 起止標簽的問題。即:

<?php // Code... ?>

通常就是上面的形式,除此之外還有一種簡寫形式:

<? /* Code... */ ?>

還可以把<?php echo $xxoo;?>簡寫成:<?= $xxoo;?>這種簡寫形式被稱為 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.

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; }

該函數的返回值是一個數組:

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(); }

?

與50位技術專家面對面20年技術見證,附贈技術全景圖

總結

以上是生活随笔為你收集整理的PHP5.2至5.6的新增功能详解的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。