短閉包,也叫做箭頭函數(shù),是一種用 php 編寫的短函數(shù).當(dāng)向函數(shù)中傳遞閉包時(shí),這個(gè)功能是非常有用的,比如使用 array_map
或是 array_filter
函數(shù)時(shí).
這就是它們看起來(lái)的樣子:
// Post 對(duì)象的集合
$posts = [/* … */];
$ids = array_map(fn($post) => $post->id, $posts);
而以前,你必須這樣寫:
$ids = array_map(function ($post) {
return $post->id;
}, $posts);
我們來(lái)總結(jié)一下短閉包函數(shù)如何使用.
- 在 PHP 7.4 里可用
- 以
fn
關(guān)鍵字開(kāi)頭 - 只能包含 一個(gè)表達(dá)式, 即返回表達(dá)式
return
關(guān)鍵字可忽略- 參數(shù)和返回類型均可做類型暗示
上面示例更嚴(yán)格的類型限定寫法可寫作:
$ids = array_map(fn(Post $post): int => $post->id, $posts);
有兩點(diǎn)需要提及:
- 還允許使用擴(kuò)展操作符
- 允許引用,兩個(gè)參數(shù)都可以作為返回值
假如你想要通過(guò)引用的方式返回結(jié)果,應(yīng)該使用以下語(yǔ)法:
fn&($x) => $x
簡(jiǎn)而言之,除了只允許一個(gè)表達(dá)式以外,簡(jiǎn)短的閉包和普通閉包的功能是一樣的。
單行
你應(yīng)該正確的理解它:短閉包只能有一個(gè)表達(dá)式。這意味著閉包體中不能有多行。
原因如下:短閉包的目的是為了減少冗余。當(dāng)然,在任何情況下, fn
都比 function
短。然而, RFC 的創(chuàng)建者 Nikita Popov 認(rèn)為,如果你要處理的是多行表達(dá)式的函數(shù),那么使用閉包獲得的益處就更少了。
畢竟,多行閉包的定義已經(jīng)很冗余了,所以,有和沒(méi)有這2個(gè)關(guān)鍵字( function
和 return
)將不會(huì)有太大區(qū)別。
你是否同意這個(gè)觀點(diǎn)取決于你自己。雖然我可以在我的項(xiàng)目中想到很多單行閉包的場(chǎng)景,但也有很多多行閉包的情況,從個(gè)人角度,我會(huì)喜歡這些情況下的簡(jiǎn)短語(yǔ)法。
不過(guò)還是有希望的:未來(lái)可能會(huì)添加多行短閉包,但那也是一個(gè)單獨(dú)的 RFC 。
外部作用域的值
短閉包和普通閉包的另一個(gè)顯著特征是,短閉包不需要用 use
關(guān)鍵字就能訪問(wèn)外部作用域的數(shù)據(jù)。
$modifier = 5;
array_map(fn($x) => $x * $modifier, $numbers);
需要注意的是,不能修改外部范圍中的變量。因?yàn)樗鼘儆谥祩鬟f而不是引用傳遞。這意味著你可以改變短閉包內(nèi)的 $modifier
變量,但它不會(huì)對(duì)外部作用域中的 $modifier
變量產(chǎn)生影響。
當(dāng)然,有一個(gè)例外,那就是 $this
關(guān)鍵字,它的作用與普通閉包中的作用完全相同:
array_map(fn($x) => $x * $this->modifier, $numbers);
發(fā)展前景
我已經(jīng)提到過(guò)的多行閉包仍然是將來(lái)的一個(gè)發(fā)展可能.另外一個(gè)在我腦海中的想法就是在允許在類中使用短閉包,比如 getters
和 setters
函數(shù).
class Post {
private $title;
fn getTitle() => $this->title;
}
總而言之,短閉包是一個(gè)很受歡迎的特性,盡管有很多地方需要提高.其中最有可能就是多行閉包了.