PHP 7.4.0發布!一起看看有哪些新特性


PHP 7.4.0 發布了,此版本標志着 PHP 7 系列的第四次特性更新。

看了英文手冊后,發現其進行了許多改進,並帶來了一些新特性,現在將這些新特性您:

 

1.Typed Properties 類型屬性

類屬性現在支持類型聲明,以下示例將強制 $User-> id 只能分配 int 值,而 $User-> name 只能分配 string 值。

<?php

class User {

    public int $id;

    public string $name;

}

?>

● 它們自PHP 7.4起可用。

● 它們只在類中可用,並且需要訪問修飾符:public、protected、private、var。

● 除了void和callable之外,所有類型都是允許的。

PHP是我們喜歡和討厭的一種動態語言,它將強制類型轉換做的太好,有時也會引起反作用。假設您在期望整數的地方傳遞了一個字符串,PHP將嘗試自動轉換該字符串:

class Bar

{

    public int $i;

}

$bar = new Bar;

$bar->i = '1'; // 1

如果不喜歡這種行為,可以通過聲明嚴格類型來禁用它:

declare(strict_types=1);
$bar = new Bar; $bar->i = '1'; // 1
 Fatal error: Uncaught TypeError: Typed property Bar::$i must be int, string used

 

2.Arrow Functions 箭頭函數

箭頭函數提供了用於定義具有隱式按值作用域綁定的函數的簡寫語法。

<?php

$factor = 10;

$nums = array_map(fn($n) => $n * $factor, [1, 2, 3, 4]);

// $nums = array(10, 20, 30, 40);

?>

● 自PHP 7.4起可用

● 他們以fn關鍵字開頭

● 它們只能有一個表達式,即return語句

● 不允許return關鍵字

● 參數和返回類型可以是類型提示

您沒看錯:短閉包只能有一個表達式。這意味着您不能包含多行。

理由如下:

簡短閉包的目標是減少冗長。fn當然在所有情況下都比function短。然而,如果您處理的是多行函數,那么使用短閉包所獲得的好處就更少。

畢竟,按照定義,多行閉包已經更加冗長;因此能夠跳過兩個關鍵字(function和return)不會有太大的區別。

 

3.Limited Return Type Covariance and Argument Type Contravariance 有限返回類型協變與參數類型逆變

僅當使用自動加載時,才提供完全協變/逆變支持。在單個文件中,只能使用非循環類型引用,因為所有類在被引用之前都必須可用。

<?php

class A {}

class B extends A {}

class Producer {

    public function method(): A {}

}

class ChildProducer extends Producer {

    public function method(): B {}

}

?>

 

4.Unpacking Inside Arrays 打包內部數組

<?php

$parts = ['apple', 'pear'];

$fruits = ['banana', 'orange', ...$parts, 'watermelon'];

// ['banana', 'orange', 'apple', 'pear', 'watermelon'];

?>

 

5.Numeric Literal Separator 數值文字分隔符

數字文字可以在數字之間包含下划線。

<?php

6.674_083e-11; // float

299_792_458;   // decimal

0xCAFE_F00D;   // hexadecimal

0b0101_1111;   // binary

?>

限制

唯一的限制是數字文字中的每個下划線必須直接位於兩個數字之間。這條規則意味着下面的用法都不是有效的數字文字:

_100; 

100_;       

1__1;       

1_.0; 1._0; 

0x_123;     

0b_101;     

1_e2; 1e_2; 

PHP功能不受影響

在數字文字的數字之間添加下划線不會改變其值。下划線在詞法分析階段被刪除,因此運行時不受影響。

var_dump(1_000_000); // int(1000000)

 

6.Weak References 弱引用

弱引用可以保留對對象的引用,不會阻止對象被銷毀。

弱引用允許保留對對象的引用,而該對象不會阻止對象被銷毀;它們對於實現類似緩存的結構非常有用。

原則上,弱引用對象並不復雜,只需要(ab)使用Zend或下面的層,因為我們不直接支持它。

final class WeakReference {

    public static function create(object $object) : WeakReference;
  
    public function get() : ?object;

}

 

7.Allow Exceptions from __toString() 允許從 __toString() 拋出異常

現在允許從 __toString() 引發異常,以往這會導致致命錯誤,字符串轉換中現有的可恢復致命錯誤已轉換為 Error 異常。

7.4以前禁止從__toString()拋出異常,如果__toString()異常,將導致致命錯誤。

從技術角度來看,這種限制最終是無效的,因為字符串轉換期間的異常仍然可以由將可恢復錯誤轉換為異常的錯誤處理程序觸發:

set_error_handler(function() {

    throw new Exception();

});

try {

    (string) new stdClass;

} catch (Exception $e) {

    echo "(string) threw an exception...\n";

}

另外,將“不能轉換為字符串”和“__toString()必須返回一個字符串值”可恢復的致命錯誤轉換為正確的錯誤異常,這與PHP 7中建立的錯誤策略一致。

 

8.Opcache Preloading Opcache 預加載

新增 Opcache 預加載支持。

在PHP 7.4中,添加了對預加載的支持,這是一個可以顯著提高代碼性能的特性。

簡而言之,這是它的工作方式:

● 為了預加載文件,您需要編寫一個自定義PHP腳本

● 該腳本在服務器啟動時執行一次

● 所有預加載的文件在內存中都可用於所有請求

● 在重新啟動服務器之前,對預加載文件所做的更改不會產生任何影響

雖然預加載是建立在opcache之上的,但它並不是完全一樣的。Opcache將獲取您的PHP源文件,將其編譯為“ opcodes”,然后將這些編譯后的文件存儲在磁盤上

您可以將操作碼看作是代碼的底層表示,在運行時很容易解釋。因此,opcache會跳過源文件和PHP解釋器在運行時實際需要之間的轉換步驟。巨大的勝利!

但我們還有更多的收獲。Opcached文件不知道其他文件。如果類A是從類B擴展而來的,那么仍然需要在運行時將它們鏈接在一起。此外,opcache執行檢查以查看源文件是否被修改,並將基於此使其緩存失效。

因此,這就是預加載發揮作用的地方:它不僅將源文件編譯為操作碼,而且還將相關的類、特征和接口鏈接在一起。然后,它將這個“已編譯”的可運行代碼blob(即:PHP解釋器可以使用的代碼)保存在內存中

現在,當請求到達服務器時,它可以使用已經加載到內存中的部分代碼庫,而不會產生任何開銷。

為了進行預加載,開發人員必須告知服務器要加載哪些文件。這是用一個簡單的PHP腳本完成的,確實沒有什么困難。

規則很簡單:

● 您提供一個預加載腳本,並使用opcache.preload命令將其鏈接到您的php.ini文件中。

● 您要預加載的每個PHP文件都應該傳遞到opcache_compile_file(),或者在預加載腳本中只需要一次。

假設您想要預加載一個框架,例如Laravel。您的腳本必須遍歷vendor/laravel目錄中的所有PHP文件,並將它們一個接一個地添加。

在php.ini中:

opcache.preload=/path/to/project/preload.php

這是一個虛擬的實現:

$files = /* 要預加載的文件數組 */;

foreach ($files as $file) {

    opcache_compile_file($file);

}

有一個警告!為了預加載文件,還必須預加載它們的依賴項(接口,特征和父類)

如果類依賴項有任何問題,則會在服務器啟動時通知您:

Can't preload unlinked class

Illuminate\Database\Query\JoinClause: 

Unknown parent 

Illuminate\Database\Query\Builder

這不是一個致命的問題,您的服務器可以正常工作。但你不會得到所有你想要的預加載文件。

幸運的是,還有一種確保鏈接文件也被加載的方法:您可以使用require_once代替opcache_compile_file,讓已注冊的autoloader(可能是composer的)負責其余的工作。

$files = /* 要預加載的文件數組 */;

foreach ($files as $file) {

    require_once($file);

}

還有一些需要注意的地方。例如,如果您試圖預加載Laravel,那么框架中的一些類依賴於其他尚不存在的類。例如,文件系統緩存類\ lighting \ filesystem \ cache依賴於\League\Flysystem\Cached\Storage\AbstractCache,如果您從未使用過文件系統緩存,則可能無法將其安裝到您的項目中。

#有效嗎?

這當然是最重要的問題:所有文件都正確加載了嗎?您可以簡單地通過重新啟動服務器來測試它,然后將opcache_get_status()的輸出轉儲到PHP腳本中。您將看到它有一個名為preload_statistics的鍵,它將列出所有預加載的函數、類和腳本;以及預加載文件消耗的內存。

#性能

現在到最重要的問題:預加載真的能提高性能嗎?

答案是肯定的:我進行了一些基准測試。

有趣的是,您可以決定僅預加載代碼庫中經常使用的類。基准測試顯示,只加載大約100個熱門類,實際上可以獲得比預加載所有類更好的性能收益。預加載全部類,性能提升13%,而預加載熱門類,則提升有17%。

當然,應該預加載哪些類取決於您的項目。明智的做法是在開始時盡可能多地預加載。

此外還有一些棄用,以及從核心中刪除一些擴展,詳情查看英文原版手冊:

https://www.php.net/manual/zh/migration74.new-features.php


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM