PHP5.3, PHP5.4, PHP5.5新特性

由於用到PHP新版本,一些新特性必需要了解,且有些能夠在開發時就使用,若是不使用,那麼何須升級PHP版本呢,顯得有些得不償失了!
因此整理了一下 一些特性,有可能不全,待添加php

PHP 5.3中的新特性html

一.PHP 5.3中的新特性
1. 支持命名空間 (Namespace)
2. 支持延遲靜態綁定(Late Static Binding)
3. 支持goto語句
4. 支持閉包、Lambda/Anonymous函數
5. 新增兩個魔術方法__callStatic()和__invoke()
6. 新增Nowdoc語法
7. 在類外也可以使用const來定義常量
8. 三元運算符增長了一個快捷書寫方式:
9. HTTP狀態碼在200-399範圍內均被認爲訪問成功
10.支持動態調用靜態方法java

1.支持命名空間 (Namespace)
毫無疑問,命名空間是PHP5.3所帶來的最重要的新特性。有了命名空間的概念,在開發大型站點時,就比較容易設計出靈活的結構,同時避免不一樣包中的類名或變量名產生衝突。mysql

在PHP5.3以前,慣例的劃分Package的辦法是經過目錄名來分隔代碼文件,代碼中的類名則用下劃線_來表示目錄。例如web

1 <!--
2 <?php
3 class Zend_Db_Table_Select {}
4 // 表示當前這個類的文件位於Zend/Db/Table/Select目錄下
5 ?>
6 -->

這樣的命名方式被PEAR、Zend Framework及各類PHP項目普遍採用。雖然該方法能夠避免不一樣包或類庫中的類名產生衝突,但在書寫代碼的時候顯得較爲麻煩和笨拙。
在PHP5.3中,則只須要指定不一樣的命名空間便可,命名空間的分隔符爲反斜杆\。
select.php正則表達式

1 <!--
2 <?php
3 namespace Zend\Db\Table;
4 class Select {}
5 ?>
6 -->

這樣即便其它命名空間下存在名爲Select的類,程序在調用時也不會產生衝突。代碼的可讀性也有所增長。
調用方法
call.php算法

1 <!--
2 <?php
3 //namespace Zend\Db;
4 include('select.php');
5 $s new Zend\Db\Table\Select();
6 $s->test();
7 ?>
8 -->

2.支持延遲靜態綁定(Late Static Binding)
在PHP5中,咱們能夠在類中經過self關鍵字或者__CLASS__來判斷或調用當前類。但有一個問題,若是咱們是在子類中調用,獲得的結果將是父類。由於在繼承父類的時候,靜態成員就已經被綁定了。 例如:sql

01 <!--
02 <?php
03 class A {
04     public static function who() {
05         echo __CLASS__;
06     }
07     public static function test() {
08         self::who();
09     }
10 }
11 class extends A {
12     public static function who() {
13          echo __CLASS__;
14     }
15 }
16 B::test();
17 ?>
18 -->

以上代碼輸出的結果是:
A
這和咱們的預期不一樣,咱們原來想獲得子類的相應結果。
PHP 5.3.0中增長了一個static關鍵字來引用當前類,即實現了延遲靜態綁定:編程

01 <!--
02 <?php
03 class A {
04     public static function who() {
05         echo __CLASS__;
06     }
07     public static function test() {
08         static::who(); // 這裏實現了延遲的靜態綁定
09     }
10 }
11 class extends A {
12     public static function who() {
13          echo __CLASS__;
14     }
15
16  
17 B::test();
18 ?>
19 -->

以上代碼輸出的結果是:
B
3.支持goto語句
多數計算機程序設計語言中都支持無條件轉向語句goto,當程序執行到goto語句時,即轉向由goto語句中的標號指出的程序位置繼續執行。儘管goto語句有可能會致使程序流程不清晰,可讀性減弱,但在某些狀況下具備其獨特的方便之處,例如中斷深度嵌套的循環和 if 語句。json

01 <!--
02 <?php
03 goto a;
04 echo 'Foo';
05 a:
06 echo 'Bar';
07 for($i=0,$j=50; $i<100; $i++) {
08   while($j--) {
09     if($j==17) goto end;
10   }
11 }
12 echo "i = $i";
13 end:
14 echo 'j hit 17';
15 ?>
16 -->

4.支持閉包、Lambda/Anonymous函數
閉包(Closure)函數和Lambda函數的概念來自於函數編程領域。例如JavaScript 是支持閉包和 lambda 函數的最多見語言之一。

在PHP中,咱們也能夠經過create_function()在代碼運行時建立函數。但有一個問題:建立的函數僅在運行時才被編譯,而不與其它代碼同時被編譯成執行碼,所以咱們沒法使用相似APC這樣的執行碼緩存來提升代碼執行效率。

在PHP5.3中,咱們可使用Lambda/匿名函數來定義一些臨時使用(即用即棄型)的函數,以做爲array_map()/array_walk()等函數的回調函數。

01 <!--
02 <?php
03 echo preg_replace_callback('~-([a-z])~'function ($match) {
04     return strtoupper($match[1]);
05 }, 'hello-world');
06 // 輸出 helloWorld
07 $greet function($name)
08 {
09     printf("Hello %s\r\n"$name);
10 };
11 $greet('World');
12 $greet('PHP');
13 //...在某個類中
14 $callback =      function ($quantity$productuse ($tax, &$total)         {
15    $pricePerItem = constant(__CLASS__ "::PRICE_" .  strtoupper($product));
16    $total += ($pricePerItem $quantity) * ($tax + 1.0);
17  };
18 array_walk($products$callback);
19 ?>
20 -->

5. 新增兩個魔術方法__callStatic()和__invoke()
PHP中本來有一個魔術方法__call(),當代碼調用對象的某個不存在的方法時該魔術方法會被自動調用。新增的__callStatic()方法則只用於靜態類方法。當嘗試調用類中不存在的靜態方法時,__callStatic()魔術方法將被自動調用。

01 <!--
02 <?php
03 class MethodTest {
04     public function __call($name$arguments) {
05         // 參數 $name 大小寫敏感
06         echo "調用對象方法 '$name' "
07              . implode(' -- '$arguments). "\n";
08     
09  
10     /**  PHP 5.3.0 以上版本中本類方法有效  */
11     public static function __callStatic($name$arguments) {
12         // 參數 $name 大小寫敏感
13         echo "調用靜態方法 '$name' "
14              . implode(' -- '$arguments). "\n";
15     }
16
17  
18 $obj new MethodTest;
19 $obj->runTest('經過對象調用'); 
20  
21 MethodTest::runTest('靜態調用');  // As of PHP 5.3.0
22 ?>
23 -->

以上代碼執行後輸出以下:
調用對象方法’runTest’ –- 經過對象調用調用靜態方法’runTest’ –- 靜態調用
以函數形式來調用對象時,__invoke()方法將被自動調用。

01 <!--
02 <?php
03 class MethodTest {
04     public function __call($name$arguments) {
05         // 參數 $name 大小寫敏感
06         echo "Calling object method '$name' "
07              . implode(', '$arguments). "\n";
08     
09  
10     /**  PHP 5.3.0 以上版本中本類方法有效  */
11     public static function __callStatic($name$arguments) {
12         // 參數 $name 大小寫敏感
13         echo "Calling static method '$name' "
14              . implode(', '$arguments). "\n";
15     }
16
17  
18 $obj new MethodTest;
19 $obj->runTest('in object context'); 
20  
21 MethodTest::runTest('in static context');  // As of PHP 5.3.0
22 ?>
23 -->

6.新增Nowdoc語法
用法和Heredoc相似,但使用單引號。Heredoc則須要經過使用雙引號來聲明。
Nowdoc中不會作任何變量解析,很是適合於傳遞一段PHP代碼。

01 <!--
02 <?php
03 // Nowdoc 單引號 PHP 5.3以後支持
04 $name 'MyName';
05 echo <<<'EOT'
06 My name is "$name".
07 EOT;
08 //上面代碼輸出 My name is "$name". ((其中變量不被解析)
09 // Heredoc不加引號
10 echo <<<FOOBAR
11 Hello World!
12 FOOBAR;
13 //或者 雙引號 PHP 5.3以後支持
14 echo <<<"FOOBAR"
15 Hello World!
16 FOOBAR;
17 ?>
18 -->

支持經過Heredoc來初始化靜態變量、類成員和類常量。

01 <!--
02 <?php
03 // 靜態變量
04 function foo()
05 {
06     static $bar = <<<LABEL
07 Nothing in here...
08 LABEL;
09
10  
11 // 類成員、常量
12 class foo
13 {
14     const BAR = <<<FOOBAR
15 Constant example
16 FOOBAR; 
17  
18     public $baz = <<<FOOBAR
19 Property example
20 FOOBAR;
21 }
22 ?>
23 -->

7. 在類外也可以使用const來定義常量
PHP中定義常量一般是用這種方式:

1 <!--
2 <?php
3 define("CONSTANT""Hello world.");
4 ?>
5 -->

而且新增了一種常量定義方式:

1 <!--
2 <?php
3 const CONSTANT = 'Hello World';
4 ?>
5 -->

8. 三元運算符增長了一個快捷書寫方式

1 <!--
2 ?:
3 -->

本來格式爲是(expr1) ? (expr2) : (expr3)
若是expr1結果爲True,則返回expr2的結果。

PHP5.3新增一種書寫方式,能夠省略中間部分,書寫爲expr1 ?: expr3
若是expr1結果爲True,則返回expr1的結果
9. HTTP狀態碼在200-399範圍內均被認爲訪問成功
10.支持動態調用靜態方法

01 <!--
02 <?php
03 class Test{
04     public static function testgo()
05     {
06          echo "gogo!";
07     }
08 }
09 $class 'Test';
10 $action 'testgo';
11 $class::$action();  //輸出 "gogo!"
12 ?>
13 -->

11. 支持嵌套處理異常(Exception)
12. 新的垃圾收集器(GC),並默認啓用

二.PHP5.3中其它值得注意的改變
1. 修復了大量bug
2. PHP性能提升
3. php.ini中可以使用變量
4. mysqlnd進入核心擴展 理論上說該擴展訪問mysql速度會較以前的MySQL 和 MySQLi 擴展快(參見http://dev.mysql.com/downloads/connector/php-mysqlnd/)
5. ext/phar、ext/intl、ext/fileinfo、ext/sqlite3和ext/enchant等擴展默認隨PHP綁定發佈。其中Phar可用於打包PHP程序,相似於Java中的jar機制。
6. ereg 正則表達式函數 再也不默承認用,請使用速度更快的PCRE 正則表達式函數


PHP 5.4中的新特性

1. Buid-in web server內置了一個簡單的Web服務器
把當前目錄做爲Root Document只須要這條命令便可:
# php -S localhost:3300
也能夠指定其它路徑:
# php -S localhost:3300 -t /path/to/root
還能夠指定路由:
# php -S localhost:3300 router.php

2.Traits
Traits提供了一種靈活的代碼重用機制,即不像interface同樣只能定義方法但不能實現,又不能像class同樣只能單繼承。至於在實踐中怎樣使用,還須要深刻思考。
魔術常量爲__TRAIT__

3. Short array syntax 數組簡短語法

1 <!--
2 $arr = [1,'tsing', 'tsingpost.com'];
3 $array = [
4   "foo" => "bar",
5   "bar" => "foo"
6   ];
7 -->

4. Array dereferencing 數組值

01 <!--
02 function myfunc() {
03     return array(1,'tsing', 'tsingpost.com');
04 }
05 我認爲比數組簡短語法更方便的是dereferencing,之前咱們須要這樣:
06 $arr = myfunc();
07 echo $arr[1];
08 在PHP5.4中這樣就好了:
09 echo myfunc()[1];
10 其餘:
11 $name = explode(",", "tsings,male")[0];
12 explode(",", "tsings,male")[3] = "phper";
13 -->

5. Upload progress

Session提供了上傳進度支持,經過$_SESSION["upload_progress_name"]就能夠得到當前文件上傳的進度信息,結合Ajax就能很容易實現上傳進度條了。

6. JsonSerializable Interface
實現了JsonSerializable接口的類的實例在json_encode序列化的以前會調用jsonSerialize方法,而不是直接序列化對象的屬性。

7. Use mysqlnd by default

1 <!--
2 如今mysql, mysqli, pdo_mysql默認使用mysqlnd本地庫,在PHP5.4之前須要:
3 $./configure --with-mysqli=mysqlnd
4 如今:
5 $./configure --with-mysqli
6 -->

8.實例化類

1 <!--
2 class test{
3     function show(){
4     return 'test';
5     }
6 }
7 echo (new test())->show();
8 -->

9.支持 Class::{expr}() 語法

1 <!--
2 foreach ([new Human("Gonzalo"), new Human("Peter")] as $human) {
3     echo $human->{'hello'}();
4 }
5 -->

10.Callable typehint

01 <!--
02 function foo(callable $callback) {
03 }
04   則:
05   foo("false"); //錯誤,由於false不是callable類型
06   foo("printf"); //正確
07   foo(function(){}); //正確
08 class A {
09   static function show() {
10     }
11 }
12   foo(array("A", "show")); //正確
13 -->

11.函數類型提示的加強

01 <!--
02 因爲php是弱類型的語言,所以在php 5.0後,引入了函數類型提示的功能,其含義爲對於傳入函數中的參數都進行類型檢查,舉個例子,有以下的類:
03 class bar {
04 function foo(bar $foo) {
05 }
06 //其中函數foo中的參數規定了傳入的參數必須爲bar類的實例,不然系統會判斷出錯。一樣對於數組來講,也能夠進行判斷,好比:
07 function foo(array $foo) {
08 }
09 }
10   foo(array(1, 2, 3)); // 正確,由於傳入的是數組
11   foo(123); // 不正確,傳入的不是數組
12 -->

12.新增長了$_SERVER["REQUEST_TIME_FLOAT"],這個是用來統計服務請求時間的,並用ms來表示

1 <!--
2 echo "腳本執行時間 ", round(microtime(true) - $_SERVER["REQUEST_TIME_FLOAT"], 2), "s";
3  -->

13. 讓Json更懂中文(JSON_UNESCAPED_UNICODE)

1 <!--
2 echo json_encode("中文", JSON_UNESCAPED_UNICODE);
3 //"中文"
4 -->

14. 二進制直接量(binary number format)

1 <!--
2 $bin  = 0b1101;
3 echo $bin;
4 //13
5 -->

PHP 5.4.0 性能大幅提高, 修復超過100個bug.
廢除了register_globals, magic_quotes以及安全模式。
另外值得一提的是多字節支持已經默認啓用了,
default_charset從ISO-8859-1已經變爲UTF-8.
默認發送「Content-Type: text/html; charset=utf-8」,
你不再須要在HTML裏寫meta tag,也無需爲UTF-8兼容而傳送額外的header了。

刪除的特性
最後,咱們集中整理了幾年來標記爲已棄用的多個特性。這些特性包括 allow_call_time_pass_reference、define_syslog_variables、highlight.bg、register_globals、register_long_arrays、magic_quotes、safe_mode、zend.ze1_compatibility_mode、session.bug_compat4二、session.bug_compat_warn 以及 y2k_compliance。
除了這些特性以外,magic_quotes 多是最大的危險。在早期版本中,未考慮因 magic_quotes 出錯致使的後果,簡單編寫且未採起任何舉措使自身免受 SQL 注入攻擊的應用程序都經過 magic_quotes 來保護。若是在升級到 PHP 5.4 時未驗證已採起正確的 SQLi 保護措施,則可能致使安全漏洞。

其餘改動和特性
有一種新的「可調用的」類型提示,用於某方法採用回調做爲參數的狀況。
htmlspecialchars() 和 htmlentities() 如今可更好地支持亞洲字符,若是未在 php.ini 文件中顯式設置 PHP default_charset,這兩個函數默認使用 UTF-8 而不是 ISO-8859-1。
會話 ID 如今默認經過 /dev/urandom(或等效文件)中的熵生成,而不是與早期版本同樣成爲必須顯式啓用的一個選項。
mysqlnd 這一捆綁的 MySQL 原生驅動程序庫如今默認用於與 MySQL 通訊的各類擴展,除非在編譯時經過 ./configure 被顯式覆蓋。
可能還有 100 個小的改動和特性。從 PHP 5.3 升級到 5.4 應該極爲順暢,但請閱讀遷移指南加以確保。若是您從早期版本升級,執行的操做可能稍多一些。請查看之前的遷移指南再開始升級。


PHP 5.5中的新特性

新特性及提案列表 都至關大,並且不是按重要性排序。因此,若是你不想通讀一遍的話,這裏有四個特色我我的是最興奮的:
:一個簡單的密碼散列API
:標量類型提示
:Getter和Setter   
:生成器
如今,讓咱們來看看PHP5.5 可能會新增的功能:

一、放棄對Windows XP和2003 的支持

2.棄用e修飾符
e修飾符是指示preg_replace函數用來評估替換字符串做爲PHP代碼,而不僅是僅僅作一個簡單的字符串替換。不出所料,這種行爲會源源不斷的出現安全問題。這就是爲何在PHP5.5 中使用這個修飾符將拋出一個棄用警告。做爲替代,你應該使用preg_replace_callback函數。你能夠從RFC找到更多關於這個變化相應的信息。

3.新增函數和類

01 <!--
02 boolval()
03 PHP已經實現了strval、intval和floatval的函數。爲了達到一致性將添加boolval函數。它徹底能夠做爲一個布爾值計算,也能夠做爲一個回調函數。  
04  
05 hash_pbkdf2()
06 PBKDF2全稱「Password-Based Key Derivation Function 2」,正如它的名字同樣,是一種從密碼派生出加密密鑰的算法。這就須要加密算法,也能夠用於對密碼哈希。更普遍的說明和用法示例 
07  
08 array_column()
09 $userNames = array_column($users, 'name');
10 // is the same as
11 $userNames = [];
12 foreach ($users as $user) {
13     $userNames[] = $user['name'];
14
15  
16 intl 擴展
17 將有許多改進 intl的擴展。例如,將會有新的IntlCalendar,IntlGregorianCalendar,IntlTimeZone,IntlBreakIterator,IntlRuleBasedBreakIterator,IntlCodePointBreakIterator類。以前,我居然不知道有這麼多關於intl擴展,若是你想知道更多,我建議你去最新公告裏找 Calendar和 BreakIterator。
18 -->

4.一個簡單的密碼散列API

01 <!--
02 $password = "foo";
03 // creating the hash
04 $hash = password_hash($password, PASSWORD_BCRYPT);
05 // verifying a password
06 if (password_verify($password, $hash)) {
07     // password correct!
08 } else {
09     // password wrong!
10 }
11 -->

5.新的語言特性和加強功能。

01 <!--
02 常量引用
03 「常量引用」意味着數組能夠直接操做字符串和數組字面值。舉兩個例子:
04 function randomHexString($length) {
05     $str = '';
06     for ($i = 0; $i < $length; ++$i) {
07         $str .= "0123456789abcdef"[mt_rand(0, 15)]; // direct dereference of string
08     }
09 }
10 function randomBool() {
11     return [false, true][mt_rand(0, 1)]; // direct dereference of array
12 }
13 我不認爲在實踐中會使用此功能,但它使語言更加一致。請參閱 RFC。
14 -->

6.調用empty()函數(和其餘表達式)一塊兒工做
目前,empty()語言構造只能用在變量,而不能在其餘表達式。
在特定的代碼像empty($this->getFriends())將會拋出一個錯誤。做爲PHP5.5 這將成爲有效的代碼

7.獲取完整類別名稱

1 <!--
2 PHP5.3 中引入命名空間的別名類和命名空間短版本的功能。雖然這並不適用於字符串類名稱
3 use Some\Deeply\Nested\Namespace\FooBar;
4 // does not work, because this will try to use the global FooBar class
5 $reflection = new ReflectionClass('FooBar');
6 echo FooBar::class;
7 爲了解決這個問題採用新的FooBar::class語法,它返回類的完整類別名稱
8  -->

8.參數跳躍

1 <!--
2 若是你有一個函數接受多個可選的參數,目前沒有辦法只改變最後一個參數,而讓其餘全部參數爲默認值。
3 RFC上的例子,若是你有一個函數以下:
4 function create_query($where, $order_by, $join_type='', $execute = false, $report_errors = true) { ... }
5 那麼有沒有辦法設置$report_errors=false,而其餘兩個爲默認值。爲了解決這個跳躍參數的問題而提出:
6 create_query("deleted=0", "name", default, default, false);
7 我我的不是特別喜歡這個提議。在個人眼睛裏,代碼須要這個功能,只是設計不當。函數不該該有12個可選參數。
8  -->

9.標量類型提示

01 <!--
02 標量類型提示本來計劃進入5.4,但因爲缺少共識而沒有作。獲取更多關於爲何標量類型提示沒有作進PHP的信息,請參閱: 標量類型提示比你認爲的更難。
03 對於PHP5.5 而言,針對標量類型提示討論又一次出現,我認爲這是一個至關不錯的 提議。
04 它須要經過輸入值來指定類型。例如:123,123.0,「123」都是一個有效的int參數輸入,但「hello world」就不是。這與內部函數的行爲一致。
05 function foo(int $i) { ... }
06 foo(1);      // $i = 1
07 foo(1.0);    // $i = 1
08 foo("1");    // $i = 1
09 foo("1abc"); // not yet clear, maybe $i = 1 with notice
10 foo(1.5);    // not yet clear, maybe $i = 1 with notice
11 foo([]);     // error
12 foo("abc");  // error
13 -->


11.生成器

01 <!--
02 目前,自定義迭代器不多使用,由於它們的實現,須要大量的樣板代碼。生成器解決這個問題,並提供了一種簡單的樣板代碼來建立迭代器。
03 例如,你能夠定義一個範圍函數做爲迭代器:
04 <?php
05 function *xrange($start, $end, $step = 1) {
06     for ($i = $start; $i < $end; $i += $step) {
07         yield $i;
08     }
09 }
10 foreach (xrange(10, 20) as $i) {
11     // ...
12 }
13 上述xrange函數具備與內建函數相同的行爲,但有一點區別:不是返回一個數組的全部值,而是返回一個迭代器動態生成的值。
14  -->

12.列表解析和生成器表達式

01 <!--
02 列表解析提供一個簡單的方法對數組進行小規模操做:
03 $firstNames = [foreach ($users as $user) yield $user->firstName];
04 上述列表解析相等於下面的代碼:
05 $firstNames = [];
06 foreach ($users as $user) {
07     $firstNames[] = $user->firstName;
08 }
09 也能夠這樣過濾數組:
10 $underageUsers = [foreach ($users as $user) if ($user->age < 18) yield $user];
11 生成器表達式也很相似,可是返回一個迭代器(用於動態生成值)而不是一個數組。
12 -->

13.finally關鍵字

1 <!--
2 這個和java中的finally同樣,經典的try ... catch ... finally 三段式異常處理。
3  -->

14.foreach 支持list()

01 <!--
02 對於「數組的數組」進行迭代,以前須要使用兩個foreach,如今只須要使用foreach + list了,可是這個數組的數組中的每一個數組的個數須要同樣。看文檔的例子一看就明白了。
03 $array = [
04     [1, 2],
05     [3, 4],
06 ];
07 foreach ($array as list($a, $b)) {
08     echo "A: $a; B: $b\n";
09 }
10  -->

15.增長了opcache擴展
使用opcache會提升php的性能,你能夠和其餘擴展同樣靜態編譯(–enable-opcache)或者動態擴展(zend_extension)加入這個優化項。

16.非變量array和string也能支持下標獲取了

1 <!--
2 echo array(1, 2, 3)[0];
3 echo [1, 2, 3][0];
4 echo "foobar"[2];
5 -->
相關文章
相關標籤/搜索