PHP8引入了一些重大變更,有哪些新特性與效能優化?

Laravel00發表於2021-02-25

文章來自:mp.weixin.qq.com/s/iBw-Jfz0b2tOn63...
小編整理了一份PDF文件:【PHP8.0新特性技術詳解筆記



PHP 8.0.0 目前是最新的主要版本,它引入了一些重大變更,以及許多新特性和效能優化,PHP 8.0 值得關注的改進包括以下:
1、PHP 8.0 引入了備受期待的 Just In Time (JIT) 編譯器,能夠進一步提高 PHP 指令碼的執行速度

2、PHP 8.0 合併了諸多效能優化
3、JSON 支援現在被視為語言的核心部分,始終可用,而不是作為可選模組
4、支援 named 引數,因為它們能夠指定引數名稱而不是其確切順序
5、支援類/屬性/函式/方法/引數/常量的結構化後設資料的屬性(或在其他語言中也稱為註釋或修飾符)

6、支援可以指示多種不同型別的聯合型別,這些型別可以用作引數或函式的返回型別
7、支援靜態返回型別
8、str_contains()函式是一種檢查字串是否包含在另一個字串中的簡便方法,而不必使用strpos等。與之相似的是新的str_starts_with()和str_ends_with()函式

9、新增了Nullsafe運算子,作為在方法上應用空合併行為的快速簡便的方法
10、相比較 PHP 7.4 穩定版,PHP 8.0 在效能上大約改進了 10%,但是至少在某些方面,JIT 可以提供更多的效能。


下面我們來看看新特性和效能優化

新增 ValueError 異常

這是8新引入進來的 ValueError 的內建異常類,它繼承自 Exception 基類。你每次傳遞值到函式時候,如果檢測到是一個無效的型別時丟擲該異常,在 PHP 8 之前,這樣的操作會直接做警告處理。
示例程式碼:

<?php
declare(strict_types=1);

/**
* 傳遞陣列到 array_rand,型別正確,但是 array_rand 期望傳入的是非空陣列
* 所以會丟擲 ValueError 異常
*/
array_rand([], 0);

/**
* json_decode 的深度引數必須是有效的正整型值,
* 所以這裡也會丟擲 ValueError 異常
*/
json_decode('{}', true, -1);

執行結果:

原來 PHP 8 都正式釋出了


新增對聯合型別的支援
8新增的聯合型別,它允許一個變數擁有多個型別的值。
示例程式碼如下:

<?php
declare(strict_types=1);

/**
* 定義一個支援聯合型別的 Number 類
*/
class Number {
    private int|float $number;

    public function setNumber(int|float $number): void {
        $this->number = $number;
    }

    public function getNumber(): int|float {
        return $this->number;
    }
}

/**
* 我們可以傳遞浮點型和整型值到 Number 物件
*/
$number = new Number();
$number->setNumber(5);
var_dump($number->getNumber());

$number->setNumber(11.54);
var_dump($number->getNumber());

exit;

執行結果:

原來 PHP 8 都正式釋出了


重寫方法時允許可變引數

當你在子類重寫父類方法時,任何數量的引數都可以被替換成可變引數的,只要對應引數型別是相容的就可以。

示例程式碼如下

<?php
declare(strict_types=1);

class A {
    public function method(int $many, string $parameters, $here) {

    }
}
class B extends A {
    public function method(...$everything) {
        var_dump($everything);
    }
}

$b = new B();
$b->method('i can be overwritten!');
exit;

執行結果:

原來 PHP 8 都正式釋出了


靜態返回型別
8 中可以使用 static 關鍵字標識某個方法,且返回該方法當前所屬的類,即使它是繼承的,可用於後期靜態繫結。示例程式碼如下:

<?php
declare(strict_types=1);

class Test {
    public function doWhatever(): static {
        // Do whatever.
        return $this;
    }
}

exit;


新增 WeakMap 特性
WeakMap 允許你建立物件到任意值的對映(這個就類似 SplObjectStorage)的同時也不會阻止作為鍵的物件被垃圾回收。要是某個物件鍵被垃圾回收了,對應鍵值對就會從集合中被移除。
這一新特性非常有用,開發者不必擔心程式碼存在記憶體洩露了。大多數 PHP 開發者可能對此不關心,但是當你在編寫長時間執行的程式時,那你就一定要提防這個問題了,比如使用 ReactPHP 進行事件驅動程式設計時。用了 WeakMap 後引用的物件,就會在失效時自動被垃圾回收。
如果你在陣列中做同樣的操作,仍然會持有該物件的引用的,但是會導致記憶體洩露。
示例程式碼如下:

<?php
declare(strict_types=1);

class FooBar {
    public WeakMap $cache;

    public function __construct() {
        $this->cache = new WeakMap();
    }

    public function getSomethingWithCaching(object $obj) {
        return $this->cache[$obj] ??= $this->computeSomethingExpensive($obj);
    }

    public function computeSomethingExpensive(object $obj) {
        var_dump("I got called");
        return rand(1, 100);
    }
}

$cacheObject = new stdClass;

$obj = new FooBar;

// "I got called" 只會列印一次
$obj->getSomethingWithCaching($cacheObject);
$obj->getSomethingWithCaching($cacheObject);

var_dump(count($obj->cache));

// 刪除該物件後 WeakMap 會釋放相應記憶體
unset($cacheObject);

var_dump(count($obj->cache));

exit;

對應的執行結果:

原來 PHP 8 都正式釋出了


變數語法調整
8的new 和 instanceof 關鍵字支援用於任意表示式了,示例程式碼如下

<?php
declare(strict_types=1);

class Foo {}
class Bar {}

$names = ['Foo', 'Bar'];
$class = new ($names[array_rand($names)]);

var_dump($class);

exit;

執行結果:

原來 PHP 8 都正式釋出了


物件的類名字面量
8 中支援使用 $object::class 獲取物件的類名,返回結果和 get_class($object) 是一樣的。示例程式碼:

<?php
declare(strict_types=1);

class Test {

}

$test = new Test();

var_dump($test::class);
var_dump(get_class($test));
exit;

執行結果:

原來 PHP 8 都正式釋出了


引數列表中允許出現可選的尾部逗號
和陣列中的尾部逗號一樣,8也支援在引數列表中定義一個尾部逗號了。示例程式碼:

<?php
declare(strict_types=1);

function method_with_many_arguments(
    $a,
    $b,
    $c,
    $d,
) {
    var_dump("this is valid syntax");
}

method_with_many_arguments(
    1,
    2,
    3,
    4,
);

exit;

上述程式碼執行結果:

原來 PHP 8 都正式釋出了


Stringable 介面

8 引入了新的 Stringable 介面,只要某個類實現了 __toString 方法,就會被當作自動實現了 Stringable 介面(這一點和 Go 介面實現有些像),而不需要顯式與宣告實現該介面,示例程式碼:

<?php
declare(strict_types=1);

class Foo {
    public function __toString() {
        return 'I am a class';
    }
}

$obj = new Foo;
var_dump($obj instanceof Stringable);

exit;

執行結果:

原來 PHP 8 都正式釋出了


throw 已經支援被用作表示式

8支援 throw 語句可以用在只允許表示式出現的地方,比如箭頭函式、合併運算子和三元運算子等:

示例程式碼

<?php
declare(strict_types=1);

$callable = fn() => throw new Exception();

$nullableValue = null;

// $value 是非空的
$value = $nullableValue ?? throw new \InvalidArgumentException();

exit;


捕獲異常而不儲存到變數

8可以編寫 catch (Exception) 程式碼來捕獲異常,但是不用將其儲存到一個變數裡:

<?php
declare(strict_types=1);

$nullableValue = null;

try {
    $value = $nullableValue ?? throw new \InvalidArgumentException();
} catch (\InvalidArgumentException) {
    var_dump("Something went wrong");
}

exit;

上述程式碼執行結果:

原來 PHP 8 都正式釋出了


PHP 8 的新增對註解的支援
註解實際上包含了多個 RFC:

https://wiki.php.net/rfc/attributes_v2
https://wiki.php.net/rfc/attribute_amendments
https://wiki.php.net/rfc/shorter_attribute_syntax
https://wiki.php.net/rfc/shorter_attribute_syntax_change

註解是 PHP 8 引入的最大新特性之一,一開始理解起來可能有點困難(如果你有 Java 基礎的話理解起來會相對簡單)。

註解允許你新增後設資料到 PHP 函式、引數、類等,這些後設資料隨後就可以通過可程式設計方式獲取到,在 PHP 7 或者更低版本中實現這樣的功能需要解析程式碼註釋塊,而通過註解可以直接訪問深度整合到 PHP 自身。
編寫一段示例程式碼方便你理解,假設你想要允許開發者新增中介軟體到控制器類/方法,使用註解,你可以這麼做,示例程式碼:

<?php
declare(strict_types=1);

// 首先,我們需要定義註解,註解本身只是一個原生的 PHP 類,並且自身被打上了註解的註釋

#[Attribute]
class ApplyMiddleware
{
    public array $middlware = [];

    public function __construct(...$middleware)
    {
        $this->middleware = $middleware;
    }
}

// 下面的語法會新增上述註解到 MyController 類,並且傳入 auth 作為引數

#[ApplyMiddleware('auth')]
class MyController
{
    public function index()
    {
    }
}

// 然後我們就可以在類中使用反射獲取所有的 ApplyMiddleware 註解並讀取給定的中介軟體引數

$reflectionClass = new ReflectionClass(MyController::class);

$attributes = $reflectionClass->getAttributes(ApplyMiddleware::class);

foreach ($attributes as $attribute) {
    $middlewareAttribute = $attribute->newInstance();
    var_dump($middlewareAttribute->middleware);
}

exit;

執行上述程式碼,列印結果:

原來 PHP 8 都正式釋出了


8新增建構函式屬性提示支援

這個新特性是一個語法簡寫,支援將屬性宣告和建構函式屬性初始化合並在一起,示例程式碼如下:

<?php
declare(strict_types=1);

class User {
    public function __construct(
        public int $id,
        public string $name,
    ) {}
}

$user = new User(1, 'Marcel');

var_dump($user->id);
var_dump($user->name);

exit;

上述程式碼執行結果:

原來 PHP 8 都正式釋出了


php8的Trait 支援定義抽象私有方法,示例程式碼如下:

<?php
declare(strict_types=1);

trait MyTrait {
    abstract private function neededByTheTrait(): string;

    public function doSomething() {
        return strlen($this->neededByTheTrait());
    }
}

class TraitUser {
    use MyTrait;

    // 支援該語法
    private function neededByTheTrait(): string { }

    // 不支援該語法 (錯誤的返回型別)
    // private function neededByTheTrait(): stdClass { }

    // 支援該語法 (非靜態方法變成了靜態方法)
    // private static function neededByTheTrait(): string { }
}

exit;


php8新增對 match 表示式支援

match 表示式和 switch 分支語句類似,不過在語義上match表示式會更加安全並且可以直接返回值:示例程式碼如下

<?php
declare(strict_types=1);

echo match (1) {
    0 => 'Foo',
    1 => 'Bar',
    2 => 'Baz',
};

exit;

上述程式碼執行結果:

原來 PHP 8 都正式釋出了


PHP 8 引入了新的名為 mixed 的型別
該型別等價於 array| bool| callable |int |float |null |object |resource |string,示例程式碼如下:

<?php
declare(strict_types=1);

function debug_function(mixed ...$data)
{
    var_dump($data);
}

debug_function(1, 'string', []);

exit;

上述程式碼執行結果:

原來 PHP 8 都正式釋出了


8新增對 命名引數 的支援

命名引數允許基於引數名稱傳遞引數到函式,而不是引數所在的位置。那麼這樣一來,函式引數就可以自解釋了且與順序無關,並且允許跳過預設值,示例程式碼如下:

<?php
declare(strict_types=1);

array_fill(start_index: 0, num: 100, value: 50);

exit;


新增對空安全運算子 ?-> 的支援
該運算子的左側評估為 null 時,整個程式碼鏈路的執行就會被終止並且整體評估為 null。但是如果要不為 null ,那就要和普通的 -> 運算子功能一樣:

<?php
declare(strict_types=1);

class User {
    public function getAddress() {}
}

$user = new User();

$country = $user?->getAddress()?->country?->iso_code;

var_dump($country);

exit;

上述程式碼執行結果:

原來 PHP 8 都正式釋出了

本作品採用《CC 協議》,轉載必須註明作者和本文連結

相關文章