新特性

標(biāo)量類(lèi)型聲明

標(biāo)量類(lèi)型聲明 有兩種模式: 強(qiáng)制 (默認(rèn)) 和 嚴(yán)格模式。 現(xiàn)在可以使用下列類(lèi)型參數(shù)(無(wú)論用強(qiáng)制模式還是嚴(yán)格模式): 字符串(string), 整數(shù) (int), 浮點(diǎn)數(shù) (float), 以及布爾值 (bool)。它們擴(kuò)充了PHP5中引入的其他類(lèi)型:類(lèi)名,接口,數(shù)組回調(diào)類(lèi)型。

<?php
// Coercive mode
function sumOfInts(int ...$ints)
{
    return 
array_sum($ints);
}

var_dump(sumOfInts(2'3'4.1));

以上例程會(huì)輸出:

int(9)

要使用嚴(yán)格模式,一個(gè) declare 聲明指令必須放在文件的頂部。這意味著嚴(yán)格聲明標(biāo)量是基于文件可配的。 這個(gè)指令不僅影響參數(shù)的類(lèi)型聲明,也影響到函數(shù)的返回值聲明(參見(jiàn) 返回值類(lèi)型聲明, 內(nèi)置的PHP函數(shù)以及擴(kuò)展中加載的PHP函數(shù))

完整的標(biāo)量類(lèi)型聲明文檔和示例參見(jiàn)類(lèi)型聲明章節(jié)。

返回值類(lèi)型聲明

PHP 7 增加了對(duì)返回類(lèi)型聲明的支持。 類(lèi)似于參數(shù)類(lèi)型聲明,返回類(lèi)型聲明指明了函數(shù)返回值的類(lèi)型??捎玫?a class="link" href="language.types.declarations.html">類(lèi)型與參數(shù)聲明中可用的類(lèi)型相同。

<?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]));

以上例程會(huì)輸出:

Array
(
    [0] => 6
    [1] => 15
    [2] => 24
)

完整的標(biāo)量類(lèi)型聲明文檔和示例可參見(jiàn) 返回值類(lèi)型聲明.

null合并運(yùn)算符

由于日常使用中存在大量同時(shí)使用三元表達(dá)式和 isset()的情況, 我們添加了null合并運(yùn)算符 (??) 這個(gè)語(yǔ)法糖。如果變量存在且值不為null, 它就會(huì)返回自身的值,否則返回它的第二個(gè)操作數(shù)。

<?php
// Fetches the value of $_GET['user'] and returns 'nobody'
// if it does not exist.
$username $_GET['user'] ?? 'nobody';
// This is equivalent to:
$username = isset($_GET['user']) ? $_GET['user'] : 'nobody';

// Coalesces can be chained: this will return the first
// defined value out of $_GET['user'], $_POST['user'], and
// 'nobody'.
$username $_GET['user'] ?? $_POST['user'] ?? 'nobody';
?>

太空船操作符(組合比較符)

太空船操作符用于比較兩個(gè)表達(dá)式。當(dāng)$a小于、等于或大于$b時(shí)它分別返回-1、0或1。 比較的原則是沿用 PHP 的常規(guī)比較規(guī)則進(jìn)行的。

<?php
// 整數(shù)
echo <=> 1// 0
echo <=> 2// -1
echo <=> 1// 1

// 浮點(diǎn)數(shù)
echo 1.5 <=> 1.5// 0
echo 1.5 <=> 2.5// -1
echo 2.5 <=> 1.5// 1
 
// 字符串
echo "a" <=> "a"// 0
echo "a" <=> "b"// -1
echo "b" <=> "a"// 1
?>

通過(guò) define() 定義常量數(shù)組

Array 類(lèi)型的常量現(xiàn)在可以通過(guò) define() 來(lái)定義。在 PHP5.6 中僅能通過(guò) const 定義。

<?php
define
('ANIMALS', [
    
'dog',
    
'cat',
    
'bird'
]);

echo 
ANIMALS[1]; // 輸出 "cat"
?>

匿名類(lèi)

現(xiàn)在支持通過(guò)new class 來(lái)實(shí)例化一個(gè)匿名類(lèi),這可以用來(lái)替代一些“用后即焚”的完整類(lèi)定義。

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

以上例程會(huì)輸出:

object(class@anonymous)#2 (0) {
}

詳細(xì)文檔可以參考 匿名類(lèi).

Unicode codepoint 轉(zhuǎn)譯語(yǔ)法

這接受一個(gè)以16進(jìn)制形式的 Unicode codepoint,并打印出一個(gè)雙引號(hào)或heredoc包圍的 UTF-8 編碼格式的字符串。 可以接受任何有效的 codepoint,并且開(kāi)頭的 0 是可以省略的。

echo "\u{aa}";
echo "\u{0000aa}";
echo "\u{9999}";

以上例程會(huì)輸出:

a
a (same as before but with optional leading 0's)
香

Closure::call()

Closure::call() 現(xiàn)在有著更好的性能,簡(jiǎn)短干練的暫時(shí)綁定一個(gè)方法到對(duì)象上閉包并調(diào)用它。

<?php
class {private $x 1;}

// PHP 7 之前版本的代碼
$getXCB = function() {return $this->x;};
$getX $getXCB->bindTo(new A'A'); // 中間層閉包
echo $getX();

// PHP 7+ 及更高版本的代碼
$getX = function() {return $this->x;};
echo 
$getX->call(new A);

以上例程會(huì)輸出:

1
1

unserialize()提供過(guò)濾

這個(gè)特性旨在提供更安全的方式解包不可靠的數(shù)據(jù)。它通過(guò)白名單的方式來(lái)防止?jié)撛诘拇a注入。

<?php

// 將所有的對(duì)象都轉(zhuǎn)換為 __PHP_Incomplete_Class 對(duì)象
$data unserialize($foo, ["allowed_classes" => false]);

// 將除 MyClass 和 MyClass2 之外的所有對(duì)象都轉(zhuǎn)換為 __PHP_Incomplete_Class 對(duì)象
$data unserialize($foo, ["allowed_classes" => ["MyClass""MyClass2"]);

// 默認(rèn)情況下所有的類(lèi)都是可接受的,等同于省略第二個(gè)參數(shù)
$data unserialize($foo, ["allowed_classes" => true]);

IntlChar

新增加的 IntlChar 類(lèi)旨在暴露出更多的 ICU 功能。這個(gè)類(lèi)自身定義了許多靜態(tài)方法用于操作多字符集的 unicode 字符。

<?php

printf
('%x'IntlChar::CODEPOINT_MAX);
echo 
IntlChar::charName('@');
var_dump(IntlChar::ispunct('!'));

以上例程會(huì)輸出:

10ffff
COMMERCIAL AT
bool(true)

若要使用此類(lèi),請(qǐng)先安裝Intl擴(kuò)展

預(yù)期

預(yù)期是向后兼用并增強(qiáng)之前的 assert() 的方法。 它使得在生產(chǎn)環(huán)境中啟用斷言為零成本,并且提供當(dāng)斷言失敗時(shí)拋出特定異常的能力。

老版本的API出于兼容目的將繼續(xù)被維護(hù),assert()現(xiàn)在是一個(gè)語(yǔ)言結(jié)構(gòu),它允許第一個(gè)參數(shù)是一個(gè)表達(dá)式,而不僅僅是一個(gè)待計(jì)算的 string或一個(gè)待測(cè)試的boolean。

<?php
ini_set
('assert.exception'1);

class 
CustomError extends AssertionError {}

assert(false, new CustomError('Some error message'));
?>

以上例程會(huì)輸出:

Fatal error: Uncaught CustomError: Some error message

關(guān)于這個(gè)特性的完整說(shuō)明,包括如何在開(kāi)發(fā)和生產(chǎn)環(huán)境中配置它,可以在assert()expectations section章節(jié)找到。

Group use declarations

從同一 namespace 導(dǎo)入的類(lèi)、函數(shù)和常量現(xiàn)在可以通過(guò)單個(gè) use 語(yǔ)句 一次性導(dǎo)入了。

<?php

// PHP 7 之前的代碼
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;

// PHP 7+ 及更高版本的代碼
use some\namespace\{ClassAClassBClassC as C};
use function 
some\namespace\{fn_afn_bfn_c};
use const 
some\namespace\{ConstAConstBConstC};
?>

生成器可以返回表達(dá)式

此特性基于 PHP 5.5 版本中引入的生成器特性構(gòu)建的。 它允許在生成器函數(shù)中通過(guò)使用 return 語(yǔ)法來(lái)返回一個(gè)表達(dá)式 (但是不允許返回引用值), 可以通過(guò)調(diào)用 Generator::getReturn() 方法來(lái)獲取生成器的返回值, 但是這個(gè)方法只能在生成器完成產(chǎn)生工作以后調(diào)用一次。

<?php

$gen 
= (function() {
    yield 
1;
    yield 
2;

    return 
3;
})();

foreach (
$gen as $val) {
    echo 
$valPHP_EOL;
}

echo 
$gen->getReturn(), PHP_EOL;

以上例程會(huì)輸出:

1
2
3

在生成器中能夠返回最終的值是一個(gè)非常便利的特性, 因?yàn)樗沟谜{(diào)用生成器的客戶(hù)端代碼可以直接得到生成器(或者其他協(xié)同計(jì)算)的返回值, 相對(duì)于之前版本中客戶(hù)端代碼必須先檢查生成器是否產(chǎn)生了最終的值然后再進(jìn)行響應(yīng)處理 來(lái)得方便多了。

Generator delegation

現(xiàn)在,只需在最外層生成其中使用 yield from, 就可以把一個(gè)生成器自動(dòng)委派給其他的生成器, Traversable 對(duì)象或者 array

<?php

function gen()
{
    yield 
1;
    yield 
2;

    yield from 
gen2();
}

function 
gen2()
{
    yield 
3;
    yield 
4;
}

foreach (
gen() as $val)
{
    echo 
$valPHP_EOL;
}

?>

以上例程會(huì)輸出:

1
2
3
4

整數(shù)除法函數(shù) intdiv()

新加的函數(shù) intdiv() 用來(lái)進(jìn)行 整數(shù)的除法運(yùn)算。

<?php

var_dump
(intdiv(103));
?>

以上例程會(huì)輸出:

int(3)

會(huì)話(huà)選項(xiàng)

session_start() 可以接受一個(gè) array 作為參數(shù), 用來(lái)覆蓋 php.ini 文件中設(shè)置的 會(huì)話(huà)配置選項(xiàng)。

在調(diào)用 session_start() 的時(shí)候, 傳入的選項(xiàng)參數(shù)中也支持 session.lazy_write 行為, 默認(rèn)情況下這個(gè)配置項(xiàng)是打開(kāi)的。它的作用是控制 PHP 只有在會(huì)話(huà)中的數(shù)據(jù)發(fā)生變化的時(shí)候才 寫(xiě)入會(huì)話(huà)存儲(chǔ)文件,如果會(huì)話(huà)中的數(shù)據(jù)沒(méi)有發(fā)生改變,那么 PHP 會(huì)在讀取完會(huì)話(huà)數(shù)據(jù)之后, 立即關(guān)閉會(huì)話(huà)存儲(chǔ)文件,不做任何修改,可以通過(guò)設(shè)置 read_and_close 來(lái)實(shí)現(xiàn)。

例如,下列代碼設(shè)置 session.cache_limiterprivate,并且在讀取完畢會(huì)話(huà)數(shù)據(jù)之后馬上關(guān)閉會(huì)話(huà)存儲(chǔ)文件。

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

preg_replace_callback_array()

在 PHP 7 之前,當(dāng)使用 preg_replace_callback() 函數(shù)的時(shí)候, 由于針對(duì)每個(gè)正則表達(dá)式都要執(zhí)行回調(diào)函數(shù),可能導(dǎo)致過(guò)多的分支代碼。 而使用新加的 preg_replace_callback_array() 函數(shù), 可以使得代碼更加簡(jiǎn)潔。

現(xiàn)在,可以使用一個(gè)關(guān)聯(lián)數(shù)組來(lái)對(duì)每個(gè)正則表達(dá)式注冊(cè)回調(diào)函數(shù), 正則表達(dá)式本身作為關(guān)聯(lián)數(shù)組的鍵, 而對(duì)應(yīng)的回調(diào)函數(shù)就是關(guān)聯(lián)數(shù)組的值。

CSPRNG Functions

新加入兩個(gè)跨平臺(tái)的函數(shù): random_bytes()random_int() 用來(lái)產(chǎn)生高安全級(jí)別的隨機(jī)字符串和隨機(jī)整數(shù)。

可以使用 list() 函數(shù)來(lái)展開(kāi)實(shí)現(xiàn)了 ArrayAccess 接口的對(duì)象

在之前版本中,list() 函數(shù)不能保證 正確的展開(kāi)實(shí)現(xiàn)了 ArrayAccess 接口的對(duì)象, 現(xiàn)在這個(gè)問(wèn)題已經(jīng)被修復(fù)。

其他特性

  • 允許在克隆表達(dá)式上訪(fǎng)問(wèn)對(duì)象成員,例如: (clone $foo)->bar()。