PHP5.5至PHP7.2 新特性整理

2020-07-16 10:06:12
一、從PHP 5.5.x 移植到 PHP 5.6.x

使用表示式定義常數

在之前的 PHP 版本中, 必須使用靜態值來定義常數,宣告屬性以及指定函數引數預設值。 現在你可以使用包括數值、字串字面量以及其他常數在內的數值表示式來 定義常數、宣告屬性以及設定函數引數預設值。

<?php
const ONE = 1;
const TWO = ONE * 2;
class C {
    const THREE = TWO + 1;
    const ONE_THIRD = ONE / self::THREE;
    const SENTENCE = 'The value of THREE is '.self::THREE;
}

現在可以通過 const 關鍵字來定義型別為 array 的常數。

<?php
const ARR = ['a', 'b'];
echo ARR[0];

使用 ... 運算子定義變長引數函數

現在可以不依賴 func_get_args(), 使用 ... 運算子 來實現 變長引數函數。

<?php
function f($req, $opt = null, ...$params) {
    // $params 是一個包含了剩餘引數的陣列
    printf('$req: %d; $opt: %d; number of params: %d'."n",
           $req, $opt, count($params));
}
f(1);
f(1, 2);
f(1, 2, 3);
f(1, 2, 3, 4);
?>

以上例程會輸出:

$req: 1; $opt: 0; number of params: 0
$req: 1; $opt: 2; number of params: 0
$req: 1; $opt: 2; number of params: 1
$req: 1; $opt: 2; number of params: 2

使用 ... 運算子進行引數展開

在呼叫函數的時候,使用 ... 運算子, 將 陣列 和 可遍歷 物件展開為函數引數。 在其他程式語言,比如 Ruby中,這被稱為連線運算子。

<?php
function add($a, $b, $c) {
    return $a + $b + $c;
}
$operators = [2, 3];
echo add(1, ...$operators);
?>

以上例程會輸出:

6

use function 以及 use const

use 運算子 被進行了擴充套件以支援在類中匯入外部的函數和常數。 對應的結構為 use functionuse const

<?php
namespace NameSpace {
    const FOO = 42;
    function f() { echo __FUNCTION__."n"; }
}
namespace {
    use const NameSpaceFOO;
    use function NameSpacef;
    echo FOO."n";
    f();
}
?>

以上例程會輸出:

42
NameSpacef

使用 hash_equals() 比較字串避免時序攻擊

二、從PHP 5.6.x 移植到 PHP 7.0.x

標量型別宣告

標量型別宣告 有兩種模式: 強制 (預設) 和 嚴格模式。 現在可以使用下列型別引數(無論用強制模式還是嚴格模式): 字串(string), 整數 (int), 浮點數 (float), 以及布林值 (bool)。

<?php
// Coercive mode
function sumOfInts(int ...$ints)
{
    return array_sum($ints);
}
var_dump(sumOfInts(2, '3', 4.1));

以上例程會輸出:

int(9)

返回值型別宣告

PHP 7 增加了對返回型別宣告的支援。 類似於引數型別宣告,返回型別宣告指明了函數返回值的型別。可用的型別與引數宣告中可用的型別相同。

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

null合併運算子

由於日常使用中存在大量同時使用三元表示式和 isset()的情況, 我們新增了null合併運算子 (??) 這個語法糖。如果變數存在且值不為NULL, 它就會返回自身的值,否則返回它的第二個運算元。

<?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';
?>

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

太空船操作符用於比較兩個表示式。當$a小於、等於或大於$b時它分別返回-1、0或1。 比較的原則是沿用 PHP 的常規比較規則進行的。

<?php
// 整數
echo 1 <=> '1'; // 0
echo 1 <=> 2; // -1
echo 2 <=> 1; // 1
// 浮點數
echo '1.50' <=> 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
?>

通過 define() 定義常數陣列

Array 型別的常數現在可以通過 define() 來定義。在 PHP5.6 中僅能通過 const 定義。

define('ANIMALS', [
    'dog',
    'cat',
    'bird'
]);
echo ANIMALS[1]; // 輸出 "cat"

Closure::call()

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

<?php
class A {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);

以上例程會輸出:

1

分組 use 宣告

從同一 namespace 匯入的類、函數和常數現在可以通過單個 use 語句 一次性匯入了。

<?php
// PHP 7 之前的程式碼
use somenamespaceClassA;
use somenamespaceClassB;
use somenamespaceClassC as C;
use function somenamespacefn_a;
use function somenamespacefn_b;
use function somenamespacefn_c;
use const somenamespaceConstA;
use const somenamespaceConstB;
use const somenamespaceConstC;
// PHP 7+ 及更高版本的程式碼
use somenamespace{ClassA, ClassB, ClassC as C};
use function somenamespace{fn_a, fn_b, fn_c};
use const somenamespace{ConstA, ConstB, ConstC};
?>

生成器可以返回表示式

此特性基於 PHP 5.5 版本中引入的生成器特性構建的。 它允許在生成器函數中通過使用 return 語法來返回一個表示式 (但是不允許返回參照值), 可以通過呼叫 Generator::getReturn() 方法來獲取生成器的返回值, 但是這個方法只能在生成器完成產生工作以後呼叫一次。

整數除法函數 intdiv()

三、從PHP 7.0.x 移植到 PHP 7.1.x

可為空(Nullable)型別

引數以及返回值的型別現在可以通過在型別前加上一個問號使之允許為空。 當啟用這個特性時,傳入的引數或者函數返回的結果要麼是給定的型別,要麼是 null 。

<?php
function testReturn(): ?string
{
    return 'elePHPant';
}
var_dump(testReturn());
function testReturn(): ?string
{
    return null;
}
var_dump(testReturn());
function test(?string $name)
{
    var_dump($name);
}
test('elePHPant');
test(null);
test();

以上例程會輸出:

string(10) "elePHPant"
NULL
string(10) "elePHPant"
NULL
Uncaught Error: Too few arguments to function test(), 0 passed in...

Void 函數

一個新的返回值型別void被引入。 返回值宣告為 void 型別的方法要麼乾脆省去 return 語句,要麼使用一個空的 return 語句。 對於 void 函數來說,NULL 不是一個合法的返回值。

<?php
function swap(&$left, &$right) : void
{
    if ($left === $right) {
        return;
    }
    $tmp = $left;
    $left = $right;
    $right = $tmp;
}
$a = 1;
$b = 2;
var_dump(swap($a, $b), $a, $b);

以上例程會輸出:

null
int(2)
int(1)
Symmetric array destructuring

短陣列語法([])現在作為list()語法的一個備選項,可以用於將陣列的值賦給一些變數(包括在foreach中)。

<?php
$data = [
    [1, 'Tom'],
    [2, 'Fred'],
];
// list() style
list($id1, $name1) = $data[0];
// [] style
[$id1, $name1] = $data[0];
// list() style
foreach ($data as list($id, $name)) {
    // logic here with $id and $name
}
// [] style
foreach ($data as [$id, $name]) {
    // logic here with $id and $name
}

類常數可見性

現在起支援設定類常數的可見性。

<?php
class ConstDemo
{
    const PUBLIC_CONST_A = 1;
    public const PUBLIC_CONST_B = 2;
    protected const PROTECTED_CONST = 3;
    private const PRIVATE_CONST = 4;
}

iterable 偽類

現在引入了一個新的被稱為iterable的偽類 (與callable類似)。 這可以被用在引數或者返回值型別中,它代表接受陣列或者實現了Traversable介面的物件。 至於子類,當用作引數時,子類可以收緊父類別的iterable型別到array 或一個實現了Traversable的物件。對於返回值,子類可以拓寬父類別的 array或物件返回值型別到iterable。

<?php
function iterator(iterable $iter) : iterable
{
    foreach ($iter as $val) {
        //
    }
}

多異常捕獲處理

一個catch語句塊現在可以通過管道字元(|)來實現多個異常的捕獲。 這對於需要同時處理來自不同類的不同異常時很有用。

<?php
try {
    // some code
} catch (FirstException | SecondException $e) {
    // handle first and second exceptions
}

list()現在支援鍵名

現在list()和它的新的[]語法支援在它內部去指定鍵名。這意味著它可以將任意型別的陣列 都賦值給一些變數(與短陣列語法類似)

<?php
$data = [
    ["id" => 1, "name" => 'Tom'],
    ["id" => 2, "name" => 'Fred'],
];
// list() style
list("id" => $id1, "name" => $name1) = $data[0];
// [] style
["id" => $id1, "name" => $name1] = $data[0];
// list() style
foreach ($data as list("id" => $id, "name" => $name)) {
    // logic here with $id and $name
}
// [] style
foreach ($data as ["id" => $id, "name" => $name]) {
    // logic here with $id and $name
}

四、從PHP 7.1.x 移植到 PHP 7.2.x

新的物件型別

這種新的物件型別, object, 引進了可用於逆變(contravariant)引數輸入和協變(covariant)返回任何物件型別。

<?php
function test(object $obj) : object
{
    return new SplQueue();
}
test(new StdClass());

允許重寫抽象方法(Abstract method)

當一個抽象類繼承於另外一個抽象類的時候,繼承後的抽象類可以重寫被繼承的抽象類的抽象方法。

abstract class A
{
    abstract function test(string $s);
}
abstract class B extends A
{
    // overridden - still maintaining contravariance for parameters and covariance for return
    abstract function test($s) : int;
}

擴充套件了引數型別

重寫方法和介面實現的引數型別現在可以省略了。不過這仍然是符合LSP,因為現在這種引數型別是逆變的。

interface A
{
    public function Test(array $input);
}
class B implements A
{
    public function Test($input){} // type omitted for $input
}

允許分組名稱空間的尾部逗號

名稱空間可以在PHP 7中使用尾隨逗號進行分組引入。

use FooBar{
    Foo,
    Bar,
    Baz,
};

以上就是PHP5.5至PHP7.2 新特性整理的詳細內容,更多請關注TW511.COM其它相關文章!