性能改善:PHP 7高達兩倍快的PHP 5.6php
顯著減小內存使用正則表達式
抽象語法樹數組
一致的64位支持安全
改進的異常層次結構session
許多轉化爲異常致命錯誤php7
安全隨機數發生器閉包
刪除舊的和不支持的SAPIs和擴展併發
空合併運算符(?)app
返回和標量類型聲明dom
匿名類
零成本斷言
生成五十萬個數組,並查詢五十萬次 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 倍
生成五十萬個數組,並查詢五十萬次 value 是否存在
<?php
$a = array();
for($i=0;$i<10000;$i++){
$a[$i] = $i;
}
foreach($a as $i)
{
array_key_exists($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 倍
示例與結果摘自鳥哥博客。以 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
有兩種模式 : 強制 ( 默認 ) 和 嚴格模式。 如今可使用下列類型參數(不管用強制模式仍是嚴格模式): 字符串 (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)
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
)
項目中存在大量同時使用三元表達式和 isset() 的狀況,新增了 null 合併運算符 (??)這個語法糖。若是變量存在且值不爲 NULL , 它就會返回自身的值,不然返回它的第二個操做數。
舊版: isset($_GET[‘id']) ? $_GET[id] : err;
新版: $_GET['id'] ?? 'err';
太空船操做符用於比較兩個表達式。當 $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
?>
<?php
define('ANIMALS', ['dog', 'cat', 'bird']);
echo ANIMALS[1]; // outputs "cat"
?>
如今支持經過 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());
這接受一個以 16 進制形式的 Unicode codepoint ,並打印出一個雙引號或 heredoc包圍的 UTF-8 編碼格式的字符串。 能夠接受任何有效的 codepoint ,而且開頭的 0 是能夠省略的。
<?php
echo 「\u{9876}」
?>
舊版輸出: \u{9876}
新版輸入:頂
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);
這個特性旨在提供更安全的方式解包不可靠的數據。它經過白名單的方式來防止潛在的代碼注入。
<?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]);
新增長的 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)
預期是向後兼用並加強以前的 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
從同一 namespace 導入的類、函數和常量如今能夠經過單個 use 語句 一次性導入了。
<?php
//PHP7 以前
use some\namespace\ClassA;
use some\namespace\ClassB;
use some\namespace\ClassC as C;
use function some\namespace\fn_a;
use function some\namespace\fn_b;
use function some\namespace\fn_c;
use const some\namespace\ConstA;
use const some\namespace\ConstB;
use const some\namespace\ConstC;
// PHP7 以後
use some\namespace\{ClassA, ClassB, ClassC as C};
use function some\namespace\{fn_a, fn_b, fn_c};
use const some\namespace\{ConstA, ConstB, ConstC};
?>
接收兩個參數做爲被除數和除數,返回他們相除結果的整數部分。
<?php
var_dump(intdiv(7, 2));
?>
輸出 int(3)
新增兩個函數 : random_bytes() and random_int(). 能夠加密的生產被保護的整數和字符串。我這蹩腳的翻譯,總之隨機數變得安全了。
andom_bytes — 加密生存被保護的僞隨機字符串
random_int — 加密生存被保護的僞隨機整數
新增了一個函數 preg_replace_callback_array() ,使用該函數可使得在使用preg_replace_callback() 函數時代碼變得更加優雅。在 PHP7 以前,回調函數會調用每個正則表達式,回調函數在部分分支上是被污染了。
如今, session_start() 函數能夠接收一個數組做爲參數,能夠覆蓋 php.ini 中session 的配置項。
好比,把 cache_limiter 設置爲私有的,同時在閱讀完 session 後當即關閉。
<?php
session_start([
'cache_limiter' => 'private',
'read_and_close' => true,
]);
?>
在 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
在生成器中能夠引入另外一個或幾個生成器,只須要寫 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
在 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)
當使用引用遍歷數組時,如今 foreach 在迭代中能更好的跟蹤變化。例如,在迭代中添加一個迭代值到數組中,參考下面的代碼:
<?php
$array = [0];
foreach ($array as &$val) {
var_dump($val);
$array[1] = 1;
}
?>
PHP5 輸出:
int(0)
PHP7 輸出:
int(0)
int(1)
含十六進制字符串再也不被認爲是數字
<?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"
被移除的函數列表以下:
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 被移除了。
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
使用相似 ASP 的標籤,以及 script 標籤來區分 PHP 代碼的方式被移除。 受到影響的標籤有: <% %> 、 <%= %> 、 <script language="php"> </script>
在不匹配的上下文中以靜態方式調用非靜態方法, 在 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
將浮點數轉換爲整數的時候,若是浮點數值太大,致使沒法以整數表達的狀況下,在以前的版本中,內部函數會直接將整數截斷,並不會引起錯誤。 在 PHP 7.0 中,若是發生這種狀況,會引起 E_WARNING 錯誤,而且返回 NULL 。
JSON 擴展已經被 JSOND 擴展取代。 對於數值的處理,有如下兩點須要注意的:第一,數值不能以點號( . )結束 (例如,數值 34. 必須寫做 34.0 或 34 )。 第二,若是使用科學計數法表示數值, e 前面必須不是點號( . ) (例如, 3.e3 必須寫做3.0e3 或 3e3 )。
在配置文件 INI 文件中,再也不支持以 # 開始的註釋行, 請使用 ; (分號)來表示註釋。 此變動適用於 php.ini 以及用 parse_ini_file() 和 parse_ini_string() 函數來處理的文件。
再也不提供 $HTTP_RAW_POST_DATA 變量。 請使用 php://input 做爲替代。
在使用 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