短閉包,也叫做箭頭函數,是一種用 php 編寫的短函數。當向函數中傳遞閉包時,這個功能是非常有用的,比如使用 array_map
或是 array_filter
函數時.
譯者注:PHP7.4 計划於今年底發布,請見 Wiki:PHP 基礎信息:發行計划
這就是它們看起來的樣子:
1 // Post 對象的集合 2 $posts = [/* … */]; 3 4 $ids = array_map(fn($post) => $post->id, $posts); 5 而以前,你必須這樣寫: 6 7 $ids = array_map(function ($post) { 8 return $post->id; 9 }, $posts);
我們來總結一下短閉包函數如何使用.
- 在 PHP 7.4 里可用
- 以
fn
關鍵字開頭 - 只能包含 一個 表達式,即返回表達式
return
關鍵字可忽略- 參數和返回類型均可做類型暗示
上面示例更嚴格的類型限定寫法可寫作:
$ids = array_map(fn(Post $post): int => $post->id, $posts);
有兩點需要提及:
- 還允許使用擴展操作符
- 允許引用,兩個參數都可以作為返回值
假如你想要通過引用的方式返回結果,應該使用以下語法:
fn&($x) => $x
簡而言之,除了只允許一個表達式以外,簡短的閉包和普通閉包的功能是一樣的。
單行
你應該正確的理解它:短閉包只能有一個表達式。這意味着閉包體中不能有多行。
原因如下:短閉包的目的是為了減少冗余。當然,在任何情況下, fn
都比 function
短。然而, RFC 的創建者 Nikita Popov 認為,如果你要處理的是多行表達式的函數,那么使用閉包獲得的益處就更少了。
畢竟,多行閉包的定義已經很冗余了,所以,有和沒有這 2 個關鍵字( function
和 return
)將不會有太大區別。
你是否同意這個觀點取決於你自己。雖然我可以在我的項目中想到很多單行閉包的場景,但也有很多多行閉包的情況,從個人角度,我會喜歡這些情況下的簡短語法。
不過還是有希望的:未來可能會添加多行短閉包,但那也是一個單獨的 RFC 。
外部作用域的值
短閉包和普通閉包的另一個顯著特征是,短閉包不需要用 use
關鍵字就能訪問外部作用域的數據。
$modifier = 5; array_map(fn($x) => $x * $modifier, $numbers);
需要注意的是,不能修改外部范圍中的變量。因為它屬於值傳遞而不是引用傳遞。這意味着你可以改變短閉包內的 $modifier
變量,但它不會對外部作用域中的 $modifier
變量產生影響。
當然,有一個例外,那就是 $this
關鍵字,它的作用與普通閉包中的作用完全相同:
array_map(fn($x) => $x * $this->modifier, $numbers);
發展前景
我已經提到過的多行閉包仍然是將來的一個發展可能。另外一個在我腦海中的想法就是在允許在類中使用短閉包,比如 getters
和 setters
函數.
1 class Post { 2 private $title; 3 4 fn getTitle() => $this->title; 5 }
總而言之,短閉包是一個很受歡迎的特性,盡管有很多地方需要提高。其中最有可能就是多行閉包了.
- 很多PHPer在進階的時候總會遇到一些問題和瓶頸,業務代碼寫多了沒有方向感,不知道該從那里入手去提升,對此我整理了一些資料,包括但不限於:分布式架構、高可擴展、高性能、高並發、服務器性能調優、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql優化、shell腳本、Docker、微服務、Nginx等多個知識點高級進階干貨需要的可以免費分享給大家,需要的加群(點擊→)677079770