array_map
或是 array_filter
函式時.
這就是它們看起來的樣子:
// Post 物件的集合
$posts = [/* … */];
$ids = array_map(fn($post) => $post->id, $posts);
複製程式碼
而以前,你必須這樣寫:
$ids = array_map(function ($post) {
return $post->id;
}, $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
函式.
class Post {
private $title;
fn getTitle() => $this->title;
}
複製程式碼
總而言之,短閉包是一個很受歡迎的特性,儘管有很多地方需要提高.其中最有可能就是多行閉包了.