PHP 團隊于2020年11月26日宣布 PHP 8 正式發布!


php8的下載地址:
https://www.php.net/downloads.php


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;
上述代碼運行結果:


以上是本文的全部內容,希望對大家的學習有幫助,也希望大家多多支持 php自學中心


版權聲明:本文內容由互聯網用戶自發貢獻,該文觀點僅代表作者本人。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。如發現本站有涉嫌抄襲侵權/違法違規的內容, 請發送郵件至 舉報,一經查實,本站將立刻刪除。