PHP 7.0.0 beta1 發佈了,在帶來了引人注目的性能提高的同時,也帶來了很多語言特性方面的改變。如下由 LCTT 翻譯自對官方的升級備註,雖然目前還不是正式發佈版,不過想必距離正式發佈的特性已經差異不大了。(本文會持續追蹤更新)php
間接變量、屬性和方法引用如今以從左到右的語義進行解釋。一些例子:mysql
$$foo['bar']['baz'] // 解釋作 ($$foo)['bar']['baz']
$foo->$bar['baz'] // 解釋作 ($foo->$bar)['baz']
$foo->$bar['baz']() // 解釋作 ($foo->$bar)['baz']()
Foo::$bar['baz']() // 解釋作 (Foo::$bar)['baz']()
要恢復之前的行爲,須要顯式地加大括號:git
${$foo['bar']['baz']}
$foo->{$bar['baz']}
$foo->{$bar['baz']}()
Foo::{$bar['baz']}()
全局關鍵字如今只接受簡單變量。像之前的github
global $$foo->bar;
如今要求以下寫法:web
global ${$foo->bar};
變量或函數調用的先後加上括號再也不有任何影響。例以下列代碼,函數調用結果以引用的方式傳給一個函數算法
function getArray() { return [1, 2, 3]; }
$last = array_pop(getArray());
// Strict Standards: 只有變量能夠用引用方式傳遞
$last = array_pop((getArray()));
// Strict Standards: 只有變量能夠用引用方式傳遞
如今不管是否使用括號,都會拋出一個嚴格標準錯誤。之前在第二種調用方式下不會有提示。sql
數組元素或對象屬性自動安裝引用順序建立,如今的結果順序將不一樣。例如:apache
$array = [];
$array["a"] =& $array["b"];
$array["b"] = 1;
var_dump($array);
如今結果是 ["a" => 1, "b" => 1],而之前的結果是 ["b" => 1, "a" => 1]。json
相關的 RFC:api
list() 再也不以反序賦值,例如:
list($array[], $array[], $array[]) = [1, 2, 3];
var_dump($array);
如今結果是 $array == [1, 2, 3] ,而不是 [3, 2, 1]。注意僅賦值順序變化了,而賦值仍然一致(LCTT 譯註:即之前的 list()行爲是從後面的變量開始逐一賦值,這樣對與上述用法就會產生 [3,2,1] 這樣的結果了。)。例如,相似以下的常規用法
list($a, $b, $c) = [1, 2, 3];
// $a = 1; $b = 2; $c = 3;
仍然保持當前的行爲。
再也不容許對空的 list() 賦值。以下全是無效的:
list() = $a;
list(,,) = $a;
list($x, list(), $y) = $a;
list() 再也不支持對字符串的拆分(之前也只在某些狀況下支持)。以下代碼:
$string = "xy";
list($x, $y) = $string;
如今的結果是: $x == null 和 $y == null (沒有提示),而之前的結果是: $x == "x" 和 $y == "y" 。此外, list() 如今老是能夠處理實現了 ArrayAccess 的對象,例如:
list($a, $b) = (object) new ArrayObject([0, 1]);
如今的結果是: $a == 0 和 $b == 1。 之前 $a 和 $b 都是 null。
相關 RFC:
foreach() 迭代再也不影響數組內部指針,數組指針可經過 current()/next() 等系列的函數訪問。例如:
$array = [0, 1, 2];
foreach ($array as &$val) {
var_dump(current($array));
}
如今將指向值 int(0) 三次。之前的輸出是 int(1)、int(2) 和 bool(false)。
在對數組按值迭代時,foreach 老是在對數組副本進行操做,在迭代中任何對數組的操做都不會影響到迭代行爲。例如:
$array = [0, 1, 2];
$ref =& $array; // Necessary to trigger the old behavior
foreach ($array as $val) {
var_dump($val);
unset($array[1]);
}
如今將打印出所有三個元素 (0 1 2),而之前第二個元素 1 會跳過 (0 2)。
在對數組按引用迭代時,對數組的修改將繼續會影響到迭代。不過,如今 PHP 在使用數字做爲鍵時能夠更好的維護數組內的位置。例如,在按引用迭代過程當中添加數組元素:
$array = [0];
foreach ($array as &$val) {
var_dump($val);
$array[1] = 1;
}
如今迭代會正確的添加了元素。如上代碼輸出是 "int(0) int(1)",而之前只是 "int(0)"。
對普通(不可遍歷的)對象按值或按引用迭代的行爲相似於對數組進行按引用迭代。這符合之前的行爲,除了如上一點所述的更精確的位置管理的改進。
對可遍歷對象的迭代行爲保持不變。
相關 RFC: https://wiki.php.net/rfc/php7_foreach
不能定義兩個同名的函數參數。例如,下面的方法將會觸發編譯時錯誤:
public function foo($a, $b, $unused, $unused) {
// ...
}
如上的代碼應該修改使用不一樣的參數名,如:
public function foo($a, $b, $unused1, $unused2) {
// ...
}
func_get_arg() 和 func_get_args() 函數再也不返回傳遞給參數的原始值,而是返回其當前值(也許會被修改)。例如:
function foo($x) {
$x++;
var_dump(func_get_arg(0));
}
foo(1);
將會打印 "2" 而不是 "1"。代碼應該改爲僅在調用 func_get_arg(s) 後進行修改操做。
function foo($x) {
var_dump(func_get_arg(0));
$x++;
}
或者應該避免修改參數:
function foo($x) {
$newX = $x + 1;
var_dump(func_get_arg(0));
}
相似的,異常回溯也再也不顯示傳遞給函數的原始值,而是修改後的值。例如:
function foo($x) {
$x = 42;
throw new Exception;
}
foo("string");
如今堆棧跟蹤的結果是:
Stack trace:
#0 file.php(4): foo(42)
#1 {main}
而之前是:
Stack trace:
#0 file.php(4): foo('string')
#1 {main}
這並不會影響到你的代碼的運行時行爲,值得注意的是在調試時會有所不一樣。
一樣的限制也會影響到 debug_backtrace() 及其它檢查函數參數的函數。
相關 RFC: https://wiki.php.net/phpng
無效的八進制表示(包含大於7的數字)如今會產生編譯錯誤。例如,下列代碼再也不有效:
$i = 0781; // 8 不是一個有效的八進制數字!
之前,無效的數字(以及無效數字後的任何數字)會簡單的忽略。之前如上 $i 的值是 7,由於後兩位數字會被悄悄丟棄。
二進制以負數鏡像位移如今會拋出一個算術錯誤:
var_dump(1 >> -1);
// ArithmeticError: 以負數進行位移
向左位移的位數超出了整型寬度時,結果老是 0。
var_dump(1 << 64); // int(0)
之前上述代碼的結果依賴於所用的 CPU 架構。例如,在 x86(包括 x86-64) 上結果是 int(1),由於其位移操做數在範圍內。
相似的,向右位移的位數超出了整型寬度時,其結果老是 0 或 -1 (依賴於符號):
var_dump(1 >> 64); // int(0)
var_dump(-1 >> 64); // int(-1)
相關 RFC: https://wiki.php.net/rfc/integer_semantics
包含十六進制數字的字符串不會再被當作數字,也不會被特殊處理。參見例子中的新行爲:
var_dump("0x123" == "291"); // bool(false) (之前是 true)
var_dump(is_numeric("0x123")); // bool(false) (之前是 true)
var_dump("0xe" + "0x1"); // int(0) (之前是 16)
var_dump(substr("foo", "0x1")); // string(3) "foo" (之前是 "oo")
// 注意:遇到了一個非正常格式的數字
filter_var() 能夠用來檢查一個字符串是否包含了十六進制數字,或這個字符串是否能轉換爲整數:
$str = "0xffff";
$int = filter_var($str, FILTER_VALIDATE_INT, FILTER_FLAG_ALLOW_HEX);
if (false === $int) {
throw new Exception("Invalid integer!");
}
var_dump($int); // int(65535)
因爲給雙引號字符串和 HERE 文檔增長了 Unicode 碼點轉義格式(Unicode Codepoint Escape Syntax), 因此帶有無效序列的 "\u{" 如今會形成錯誤:
$str = "\u{xyz}"; // 致命錯誤:無效的 UTF-8 碼點轉義序列
要避免這種狀況,須要轉義開頭的反斜槓:
$str = "\\u{xyz}"; // 正確
不過,不跟隨 { 的 "\u" 不受影響。以下代碼不會生成錯誤,和前面的同樣工做:
$str = "\u202e"; // 正確
相關 RFC:
如今有兩個異常類: Exception 和 Error 。這兩個類都實現了一個新接口: Throwable 。在異常處理代碼中的類型指示也許須要修改來處理這種狀況。
一些致命錯誤和可恢復的致命錯誤如今改成拋出一個 Error 。因爲 Error 是一個獨立於 Exception 的類,這些異常不會被已有的 try/catch 塊捕獲。
可恢復的致命錯誤被轉換爲一個異常,因此它們不能在錯誤處理裏面悄悄的忽略。部分狀況下,類型指示失敗再也不能忽略。
解析錯誤如今會生成一個 Error 擴展的 ParseError 。除了之前的基於返回值 / errorgetlast() 的處理,對某些可能無效的代碼的 eval() 的錯誤處理應該改成捕獲 ParseError 。
內部類的構造函數在失敗時老是會拋出一個異常。之前一些構造函數會返回 NULL 或一個不可用的對象。
一些 E_STRICT 提示的錯誤級別改變了。
相關 RFC:
靜態調用一個不兼容的 $this 上下文的非靜態調用的作法再也不支持。這種狀況下,$this 是沒有定義的,可是對它的調用是容許的,並帶有一個廢棄提示。例子:
class A {
public function test() { var_dump($this); }
}
// 注意:沒有從類 A 進行擴展
class B {
public function callNonStaticMethodOfA() { A::test(); }
}
(new B)->callNonStaticMethodOfA();
// 廢棄:非靜態方法 A::test() 不該該被靜態調用
// 提示:未定義的變量 $this
NULL
注意,這僅出如今來自不兼容上下文的調用上。若是類 B 擴展自類 A ,調用會被容許,沒有任何提示。
不能使用下列類名、接口名和特殊名(大小寫敏感):
bool
int
float
string
null
false
true
這用於 class/interface/trait 聲明、 class_alias() 和 use 語句中。
此外,下列類名、接口名和特殊名保留作未來使用,可是使用時尚不會拋出錯誤:
resource
object
mixed
numeric
yield 語句結構當用在一個表達式上下文時,再也不要求括號。它如今是一個優先級在 「print」 和 「=>」 之間的右結合操做符。在某些狀況下這會致使不一樣的行爲,例如:
echo yield -1;
// 之前被解釋以下
echo (yield) - 1;
// 如今被解釋以下
echo yield (-1);
yield $foo or die;
// 之前被解釋以下
yield ($foo or die);
// 如今被解釋以下
(yield $foo) or die;
這種狀況能夠經過增長括號來解決。
移除了 ASP (<%) 和 script (<script language=php>) 標籤。
不支持以引用的方式對 new 的結果賦值。
不支持對一個來自非兼容的 $this 上下文的非靜態方法的域內調用。細節參見: https://wiki.php.net/rfc/incompat_ctx 。
不支持 ini 文件中的 # 風格的備註。使用 ; 風格的備註替代。
$HTTP_RAW_POST_DATA 再也不可用,使用 php://input 流替代。
call_user_method() 和 call_user_method_array() 再也不存在。
在一個輸出緩衝區被建立在輸出緩衝處理器裏時, ob_start() 再也不發出 E_ERROR,而是 E_RECOVERABLE_ERROR。
改進的 zend_qsort (使用 hybrid 排序算法)性能更好,並更名爲 zend_sort。
增長靜態排序算法 zend_insert_sort。
移除 fpm-fcgi 的 dl() 函數。
setcookie() 若是 cookie 名爲空會觸發一個 WARNING ,而不是發出一個空的 set-cookie 頭。
Curl:
Date:
DBA
GMP
Intl:
libxml:
Mcrypt
Session
Opcache
OpenSSL:
PCRE:
PDO_pgsql:
Standard:
JSON:
Stream:
XSL:
Core
Opcache
OpenSSL
Reflection
Stream
Core
OpenSSL
parse_ini_string():
unserialize():
proc_open():
array_column():
stream_context_create()
GMP
PCRE:
Standard . 添加了整數除法 intdiv() 函數。 . 添加了重置錯誤狀態的 error_clear_last() 函數。
Zlib: . 添加了 deflate_init()、 deflate_add()、 inflate_init()、 inflate_add() 函數來運行遞增和流的壓縮/解壓。
(暫無)
更多細節參見:
注意:NSAPI 沒有在 RFC 中投票,不過它會在之後移除。這就是說,它相關的 SDK 從此不可用。
Core . 添加 PHP_INT_MIN
Zlib
GD
函數:
資源:
Core
ftp
odbc