文章來自: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);
執行結果:
新增對聯合型別的支援
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
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;
執行結果:
靜態返回型別
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;
對應的執行結果:
變數語法調整
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;
執行結果:
物件的類名字面量
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;
執行結果:
引數列表中允許出現可選的尾部逗號
和陣列中的尾部逗號一樣,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;
上述程式碼執行結果:
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;
執行結果:
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 的新增對註解的支援
註解實際上包含了多個 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;
執行上述程式碼,列印結果:
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;
上述程式碼執行結果:
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 引入了新的名為 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;
上述程式碼執行結果:
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;
上述程式碼執行結果:
本作品採用《CC 協議》,轉載必須註明作者和本文連結