總結PHP 7新增加的特性

Sandy發表於2019-03-04

?? 運算子(NULL 合併運算子) 把這個放在第一個說是因為我覺得它很有用。用法: $a = $_GET[`a`] ?? 1;它相當於: <?PHP $a = isset($_GET[`a`]) ? $_GET[`a`] : 1; 我們知道三元運算子是可以這樣用的: $a ?: 1但是這是建立在 $a 已經定義了的前提上。
?? 運算子(NULL 合併運算子)

把這個放在第一個說是因為我覺得它很有用。用法:

<?php 
$a = $_GET[`a`] ?? 1; 它相當於:

$a = isset($_GET[`a`]) ? $_GET[`a`] : 1; 我們知道三元運算子是可以這樣用的:

$a ?: 1 但是這是建立在 $a 已經定義了的前提上。新增的 ?? 運算子可以簡化判斷。複製程式碼

1.函式返回值型別宣告
官方文件提供的例子(注意 … 的邊長引數語法在 PHP 5.6 以上的版本中才有):

<?php
function arraysSum(array ...$arrays): array
{
    return array_map(function(array $array): int {
        return array_sum($array);
    }, $arrays);
}

print_r(arraysSum([1,2,3], [4,5,6], [7,8,9])); 從這個例子中可以看出現在函式(包括匿名函式)都可以指定返回值的型別。複製程式碼

這種宣告的寫法有些類似於 Swift:

func sayHello(personName: String) -> String {
    let greeting = "Hello, " + personName + "!"
    return greeting
}複製程式碼

這個特性可以幫助我們避免一些 PHP 的隱式型別轉換帶來的問題。在定義一個函式之前就想好預期的結果可以避免一些不必要的錯誤。

不過這裡也有一個特點需要注意。PHP 7 增加了一個 declare 指令:strict_types,既使用嚴格模式。

使用返回值型別宣告時,如果沒有宣告為嚴格模式,如果返回值不是預期的型別,PHP 還是會對其進行強制型別轉換。但是如果是嚴格模式, 則會出發一個 TypeError 的 Fatal error。

強制模式:

<?php 
function foo($a) : int { return $a; } foo(1.0); 
以上程式碼可以正常執行,foo 函式返回 int 1,沒有任何錯誤。複製程式碼

嚴格模式:

<?php 
declare(strict_types=1); 
function foo($a) : int { return $a; } foo(1.0);
PHP Fatal error: Uncaught TypeError: Return value of foo() must be of the type integer, float returned in test.php:6複製程式碼

在宣告之後,就會觸發致命錯誤。

是不是有點類似與 js 的 strict mode?

標量型別宣告
PHP 7 中的函式的形參型別宣告可以是標量了。在 PHP 5 中只能是類名、介面、array 或者 callable (PHP 5.4,即可以是函式,包括匿名函式),現在也可以使用 string、int、float和 bool 了。

官方示例:

<?php 
// Coercive mode function sumOfInts(int ...$ints) { return array_sum($ints); } var_dump(sumOfInts(2, `3`, 4.1));複製程式碼

需要注意的是上文提到的嚴格模式的問題在這裡同樣適用:強制模式(預設,既強制型別轉換)下還是會對不符合預期的引數進行強制型別轉換,嚴格模式下則觸發 TypeError 的致命錯誤。

1.use 批量宣告
PHP 7 中 use 可以在一句話中宣告多個類或函式或 const 了:

<?php 
use some
amespace{ClassA, ClassB, ClassC as C}; 
use function some
amespace{fn_a, fn_b, fn_c}; 
use const some
amespace{ConstA, ConstB, ConstC}; 
但還是要寫出每個類或函式或 const 的名稱(並沒有像 Python 一樣的 from some import * 的方法)。複製程式碼

需要留意的問題是:如果你使用的是基於 composer 和 PSR-4 的框架,這種寫法是否能成功的載入類檔案?其實是可以的,composer 註冊的自動載入方法是在類被呼叫的時候根據類的名稱空間去查詢位置,這種寫法對其沒有影響。

其他的特性
其他的一些特性我就不一一介紹了,有興趣可以檢視官方文件:php.net/manual/en/m…

簡要說幾個:

PHP 5.3 開始有了匿名函式,現在又有了匿名類了; define 現在可以定義常量陣列; 閉包( Closure)增加了一個 call 方法; 生成器(或者叫迭代器更合適)可以有一個最終返回值(return),也可以通過 yield from 的新語法進入一個另外一個生成器中(生成器委託)。 生成器的兩個新特性(return 和 yield from)可以組合。具體的表象大家可以自行測試。PHP 7 現在已經到 RC5 了,最終的版本應該會很快到來。
1、測試用例一:
生成五十萬個陣列,並查詢五十萬次key是否存在

<?php
$a = array();
for($i=0;$i<500000;$i++){ $a[$i]=”$i;” }=”” foreach($a=”” as=”” $i)=”” {=”” array_key_exists($i,=”” $a);=”” ?=””>

測試結果如下:

➜ time php test.php
php test.php
0.60s user
0.05s system
98% cpu
0.667 total
➜ time /usr/local/php7/bin/php test.php
/usr/local/php7/bin/php test.php
0.05s user
0.02s system
92% cpu
0.073 total

PHP7速度是PHP5.5的9倍
2、測試用例二:
生成五十萬個陣列,並查詢五十萬次value是否存在

<?php
$a = array();
for($i=0;$i<10000;$i++){ $a[$i]=”$i;” }=”” foreach($a=”” as=”” $i)=”” {=”” array_search($i,=”” $a);=”” ?=””>

➜ time php test.php
php test.php
0.79s user
0.01s system
99% cpu
0.809 total
➜ time /usr/local/php7/bin/php test.php
/usr/local/php7/bin/php test.php
0.08s user
0.01s system
97% cpu
0.091 total

PHP7速度是PHP5.5的8.7倍
3、測試用例三:
示例與結果摘自鳥哥部落格。以Wordpress為基礎,測試PHP7和HHVM3.2。用Apache的ab測試工具。100個併發, 10000個請求。測試前都會用100個請求預熱。
PHP7結果如下:
Concurrency Level: 100
Time taken for tests: 38.726 seconds
Complete requests: 10000
Failed requests: 0
Write errors: 0
Total transferred: 89290000 bytes
HTML transferred: 86900000 bytes
Requests per second: 258.22 [#/sec] (mean)
Time per request: 387.260 [ms] (mean)
Time per request: 3.873 [ms] (mean, across all concurrent requests)
Transfer rate: 2251.64 [Kbytes/sec] received
HHVM-3.2

HHVM結果如下:
Document Path: /wordpress/
Document Length: 8690 bytes
Concurrency Level: 100
Time taken for tests: 43.296 seconds
Complete requests: 10000
Failed requests: 0
Write errors: 0
Total transferred: 89260000 bytes
HTML transferred: 86900000 bytes
Requests per second: 230.97 [#/sec] (mean)
Time per request: 432.957 [ms] (mean)
Time per request: 4.330 [ms] (mean, across all concurrent requests)
Transfer rate: 2013.31 [Kbytes/sec] received

PHP7 – 258.22 QPS HHVM – 230.97 QPS
四、新特性

1、標量型別宣告
有兩種模式:強制(預設)和嚴格模式。現在可以使用下列型別引數(無論用強制模式還是嚴格模式):字串(string),整數(int),浮點數(float),以及布林值(bool)。它們擴充了PHP5中引入的其他型別:類名,介面,陣列和回撥型別。在舊版中,函式的引數申明只能是(Array $arr)、(CLassName $obj)等,基本型別比如Int,String等是不能夠被申明的

<?php
function check(int $bool){
var_dump($bool);
}
check(1);
check(true);
?>

若無強制型別轉換,會輸入int(1)bool(true)。轉換後會輸出bool(true) bool(true)
2、返回值型別宣告
PHP 7增加了對返回型別宣告的支援。返回型別宣告指明瞭函式返回值的型別。可用的型別與引數宣告中可用的型別相同。

<?php
function arraysSum(array …$arrays): array
{
return array_map(function(array $array): int {
return array_sum($array);
}, $arrays);
}
print_r(arraysSum([1,2,3], [4,5,6], [7,8,9]));

以上例程會輸出:
Array
(
[0] => 6
[1] => 15
[2] => 24
)
3、null合併運算子
專案中存在大量同時使用三元表示式和isset()的情況,新增了null合併運算子(??)這個語法糖。如果變數存在且值不為NULL,它就會返回自身的值,否則返回它的第二個運算元。
舊版:isset($_GET[‘id`]) ? $_GET[id] : err;
新版:$_GET[`id`] ?? `err`;
4、太空船操作符(組合比較符)
太空船操作符用於比較兩個表示式。當$a大於、等於或小於$b時它分別返回-1、0或1。比較的原則是沿用PHP的常規比較規則進行的。

<?php
// Integers
echo 1 <=> 1; // 0
echo 1 <=> 2; // -1
echo 2 <=> 1; // 1
// Floats
echo 1.5 <=> 1.5; // 0
echo 1.5 <=> 2.5; // -1
echo 2.5 <=> 1.5; // 1
// Strings
echo “a” <=> “a”; // 0
echo “a” <=> “b”; // -1
echo “b” <=> “a”; // 1
?>

5、通過define()定義常量陣列

<?php
define(`ANIMALS`, [`dog`, `cat`, `bird`]);
echo ANIMALS[1]; // outputs “cat”
?>

6、匿名類
現在支援通過new class來例項化一個匿名類,這可以用來替代一些“用後即焚”的完整類定義。

<?php
interface Logger {
public function log(string $msg);
}
class Application {
private $logger;
public function getLogger(): Logger {
return $this->logger;
}
public function setLogger(Logger $logger) {
$this->logger = $logger;
}
}
$app = new Application;
$app->setLogger(new class implements Logger {
public function log(string $msg) {
echo $msg;
}
});
var_dump($app->getLogger());

7、Unicode codepoint轉譯語法
這接受一個以16進位制形式的Unicode codepoint,並列印出一個雙引號或heredoc包圍的UTF-8編碼格式的字串。可以接受任何有效的codepoint,並且開頭的0是可以省略的。

<?php
echo “u{9876}”
?>

       舊版輸出:u{9876}
       新版輸入:頂複製程式碼

8、Closure::call()
Closure::call()現在有著更好的效能,簡短幹練的暫時繫結一個方法到物件上閉包並呼叫它。

<?php
class Test{public $name = “lixuan”;}

//PHP7和PHP5.6都可以
$getNameFunc = function(){return $this->name;};
$name = $getNameFunc->bindTo(new Test, `Test`);
echo $name();
//PHP7可以,PHP5.6報錯
$getX = function() {return $this->name;};
echo $getX->call(new Test);

9、為unserialize()提供過濾
這個特性旨在提供更安全的方式解包不可靠的資料。它通過白名單的方式來防止潛在的程式碼注入。

<?php
//將所有物件分為PHP_Incomplete_Class物件
$data = unserialize($foo, [“allowed_classes” => false]);
//將所有物件分為
PHP_Incomplete_Class物件除了ClassName1和ClassName2
$data = unserialize($foo, [“allowed_classes” => [“ClassName1”, “ClassName2”]);
//預設行為,和unserialize($foo)相同
$data = unserialize($foo, [“allowed_classes” => true]);

10、IntlChar
新增加的IntlChar類旨在暴露出更多的ICU功能。這個類自身定義了許多靜態方法用於操作多字符集的unicode字元。Intl是Pecl擴充套件,使用前需要編譯進PHP中,也可apt-get/yum/port install php5-intl

<?php
printf(`%x`, IntlChar::CODEPOINT_MAX);
echo IntlChar::charName(`@`);
var_dump(IntlChar::ispunct(`!`));
?>

以上例程會輸出:
10ffff
COMMERCIAL AT
bool(true)
11、預期
預期是向後兼用並增強之前的assert()的方法。它使得在生產環境中啟用斷言為零成本,並且提供當斷言失敗時丟擲特定異常的能力。老版本的API出於相容目的將繼續被維護,assert()現在是一個語言結構,它允許第一個引數是一個表示式,而不僅僅是一個待計算的string或一個待測試的boolean。

<?php
ini_set(`assert.exception`, 1);
class CustomError extends AssertionError {}
assert(false, new CustomError(`Some error message`));
?>

以上例程會輸出:
Fatal error: Uncaught CustomError: Some error message
12、Group use declarations
從同一namespace匯入的類、函式和常量現在可以通過單個use語句一次性匯入了。

<?php
//PHP7之前
use some
amespaceClassA;
use some
amespaceClassB;
use some
amespaceClassC as C;
use function some
amespacefn_a;
use function some
amespacefn_b;
use function some
amespacefn_c;
use const some
amespaceConstA;
use const some
amespaceConstB;
use const some
amespaceConstC;
// PHP7之後
use some
amespace{ClassA, ClassB, ClassC as C};
use function some
amespace{fn_a, fn_b, fn_c};
use const some
amespace{ConstA, ConstB, ConstC};
?>

13、intdiv()
接收兩個引數作為被除數和除數,返回他們相除結果的整數部分。

<?php
var_dump(intdiv(7, 2));
?>

輸出int(3)
14、CSPRNG
新增兩個函式: random_bytes() and random_int().可以加密的生產被保護的整數和字串。我這蹩腳的翻譯,總之隨機數變得安全了。
andom_bytes —加密生存被保護的偽隨機字串
random_int —加密生存被保護的偽隨機整數
15、preg_replace_callback_array()
新增了一個函式preg_replace_callback_array(),使用該函式可以使得在使用preg_replace_callback()函式時程式碼變得更加優雅。在PHP7之前,回撥函式會呼叫每一個正規表示式,回撥函式在部分分支上是被汙染了。
16、Session options
現在,session_start()函式可以接收一個陣列作為引數,可以覆蓋php.ini中session的配置項。
比如,把cache_limiter設定為私有的,同時在閱讀完session後立即關閉。

<?php
session_start([
`cache_limiter` => `private`,
`read_and_close` => true,
]);
?>

17、生成器的返回值
在PHP5.5引入生成器的概念。生成器函式每執行一次就得到一個yield標識的值。在PHP7中,當生成器迭代完成後,可以獲取該生成器函式的返回值。通過Generator::getReturn()得到。

<?php
function generator() {
yield 1;
yield 2;
yield 3;
return “a”;
}
$generatorClass = (“generator”)();
foreach ($generatorClass as $val) {
echo $val.” “;
}
echo $generatorClass->getReturn();
?>

輸出為:1 2 3 a
18、生成器中引入其他生成器
在生成器中可以引入另一個或幾個生成器,只需要寫yield from functionName1

<?php
function generator1(){
yield 1;
yield 2;
yield from generator2();
yield from generator3();
}
function generator2(){
yield 3;
yield 4;
}
function generator3(){
yield 5;
yield 6;
}
foreach (generator1() as $val){
echo $val, ” “;
}
?>

輸出:1 2 3 4 5 6
五、不相容性

1、foreach不再改變內部陣列指標
在PHP7之前,當陣列通過foreach迭代時,陣列指標會移動。現在開始,不再如此,見下面程式碼。

<?php
$array = [0, 1, 2];
foreach ($array as &$val) {
var_dump(current($array));
}
?>

PHP5輸出:
int(1)
int(2)
bool(false)
PHP7輸出:
int(0)
int(0)
int(0)
2、foreach通過引用遍歷時,有更好的迭代特性
當使用引用遍歷陣列時,現在foreach在迭代中能更好的跟蹤變化。例如,在迭代中新增一個迭代值到陣列中,參考下面的程式碼:

<?php
$array = [0];
foreach ($array as &$val) {
var_dump($val);
$array[1] = 1;
}
?>

PHP5輸出:
int(0)
PHP7輸出:
int(0)
int(1)
3、十六進位制字串不再被認為是數字
含十六進位制字串不再被認為是數字

<?php
var_dump(“0x123” == “291”);
var_dump(is_numeric(“0x123”));
var_dump(“0xe” + “0x1”);
var_dump(substr(“foo”, “0x1”));
?>

PHP5輸出:
bool(true)
bool(true)
int(15)
string(2) “oo”
PHP7輸出:
bool(false)
bool(false)
int(0)
Notice: A non well formed numeric value encountered in /tmp/test.php on line 5
string(3) “foo”
4、PHP7中被移除的函式
被移除的函式列表如下:
call_user_func()和call_user_func_array()從PHP 4.1.0開始被廢棄。
已廢棄的mcrypt_generic_end()函式已被移除,請使用mcrypt_generic_deinit()代替。
已廢棄的mcrypt_ecb(), mcrypt_cbc(), mcrypt_cfb()和mcrypt_ofb()函式已被移除。
set_magic_quotes_runtime(),和它的別名magic_quotes_runtime()已被移除.它們在PHP 5.3.0中已經被廢棄,並且在in PHP 5.4.0也由於魔術引號的廢棄而失去功能。
已廢棄的set_socket_blocking()函式已被移除,請使用stream_set_blocking()代替。
dl()在PHP-FPM不再可用,在CLI和embed SAPIs中仍可用。
GD庫中下列函式被移除:imagepsbbox()、imagepsencodefont()、imagepsextendfont()、imagepsfreefont()、imagepsloadfont()、imagepsslantfont()、imagepstext()
在配置檔案php.ini中,always_populate_raw_post_data、asp_tags、xsl.security_prefs被移除了。
5、new操作符建立的物件不能以引用方式賦值給變數
new操作符建立的物件不能以引用方式賦值給變數

<?php
class C {}
$c =& new C;
?>

PHP5輸出:
Deprecated: Assigning the return value of new by reference is deprecated in /tmp/test.php on line 3
PHP7輸出:
Parse error: syntax error, unexpected `new` (T_NEW) in /tmp/test.php on line 3
6、移除了ASP和script PHP標籤
使用類似ASP的標籤,以及script標籤來區分PHP程式碼的方式被移除。受到影響的標籤有:<% %>、<%= %>、
7、從不匹配的上下文發起呼叫
在不匹配的上下文中以靜態方式呼叫非靜態方法,在PHP 5.6中已經廢棄,但是在PHP 7.0中,會導致被呼叫方法中未定義$this變數,以及此行為已經廢棄的警告。

<?php
class A {
public function test() { var_dump($this); }
}
//注意:並沒有從類A繼承
class B {
public function callNonStaticMethodOfA() { A::test(); }
}
(new B)->callNonStaticMethodOfA();
?>

PHP5輸出:
Deprecated: Non-static method A::test() should not be called statically, assuming $this from incompatible context in /tmp/test.php on line 8
object(B)#1 (0) {
}
PHP7輸出:
Deprecated: Non-static method A::test() should not be called statically in /tmp/test.php on line 8
Notice: Undefined variable: this in /tmp/test.php on line 3
NULL
8、在數值溢位的時候,內部函式將會失敗
將浮點數轉換為整數的時候,如果浮點數值太大,導致無法以整數表達的情況下,在之前的版本中,內部函式會直接將整數截斷,並不會引發錯誤。在PHP 7.0中,如果發生這種情況,會引發E_WARNING錯誤,並且返回NULL。
9、JSON擴充套件已經被JSOND取代
JSON擴充套件已經被JSOND擴充套件取代。對於數值的處理,有以下兩點需要注意的:第一,數值不能以點號(.)結束(例如,數值34.必須寫作34.0或34)。第二,如果使用科學計數法表示數值,e前面必須不是點號(.)(例如,3.e3必須寫作3.0e3或3e3)。
10、INI檔案中#註釋格式被移除
在配置檔案INI檔案中,不再支援以#開始的註釋行,請使用;(分號)來表示註釋。此變更適用於php.ini以及用parse_ini_file()和parse_ini_string()函式來處理的檔案。
11、$HTTP_RAW_POST_DATA被移除
不再提供$HTTP_RAW_POST_DATA變數。請使用php://input作為替代。
12、yield變更為右聯接運算子
在使用yield關鍵字的時候,不再需要括號,並且它變更為右聯接操作符,其運算子優先順序介於print和=>之間。這可能導致現有程式碼的行為發生改變。可以通過使用括號來消除歧義。

<?php
echo yield -1;
//在之前版本中會被解釋為:
echo (yield) – 1;
//現在,它將被解釋為:
echo yield (-1);

yield $foo or die;
//在之前版本中會被解釋為:
yield ($foo or die);
//現在,它將被解釋為:
(yield $foo) or die;
?>

PHP官方網站文件:php.net/manual/zh/m…
可以瀏覽PHP5.6到PHP7時,新特性、新增函式、已經被移除的函式、不相容性、新增的類和介面等內容。

相關文章