概述
閉包和匿名函數(shù)在PHP 5.3.0中引入,這兩個特性非常有用,每個PHP開發(fā)者都應該掌握。
匿名函數(shù)其實就是沒有名稱的函數(shù),匿名函數(shù)可以賦值給變量,還能像其他任何PHP函數(shù)對象那樣傳遞。不過匿名函數(shù)仍然是函數(shù),因此可以調(diào)用,還可以傳入?yún)?shù),適合作為函數(shù)或方法的回調(diào)。
閉包是指在創(chuàng)建時封裝周圍狀態(tài)的函數(shù),即使閉包所在的環(huán)境的不存在了,閉包中封裝的狀態(tài)依然存在。
創(chuàng)建匿名函數(shù)
創(chuàng)建匿名函數(shù)很簡單:
//將匿名函數(shù)賦給一個變量,通過變量名+()的形式來調(diào)用
$greet = function () {
return "Hello World";
};
echo $greet();
結(jié)果打印:
Hello World
匿名函數(shù)和普通的PHP函數(shù)很像:常用的句法相同,也接受參數(shù),而且能返回值。不過閉包沒有函數(shù)名。
注:我們之所以能調(diào)用$greet
變量,是因為這個變量的值是一個閉包,而且閉包對象實現(xiàn)了__invoke()
魔術(shù)方法,只要變量名后有(),PHP就會查找并調(diào)用__invoke
方法。
我們通常把匿名函數(shù)當做函數(shù)或方法的回調(diào)使用,事實上,很多PHP函數(shù)都會用到匿名函數(shù),比如array_map
和preg_replace_callback
,這是使用PHP匿名函數(shù)的絕佳時機。記住,閉包和其他值一樣,可以作為參數(shù)傳入其他PHP函數(shù):
$numberPlusOne = array_map(function ($number) {
return $number += 1;
}, [1, 2, 3]);
print_r($numberPlusOne);
在匿名函數(shù)出現(xiàn)之前,要實現(xiàn)這樣的功能,PHP開發(fā)者只能單獨創(chuàng)建具名函數(shù),然后使用名稱引用這個函數(shù):
function incrementNumber ($number) {
return $number += 1;
}
$numberPlusOne = array_map(‘incrementNumber', [1, 2, 3]);
print_r($numberPlusOne);
這樣做把回調(diào)的實現(xiàn)和使用場所隔離開了,而且使用閉包實現(xiàn)代碼更加簡潔。
創(chuàng)建閉包
包含自由變量的函數(shù)與為所有這些自由變量提供了變量綁定的環(huán)境一起,被稱為閉包。
function makeHelloWorld($name) {
$i = 0;
return function()use($name, $i){
echo $name.$i. ' br>';
$i++;
};
}
$hello1 = makeHelloWorld("itbsl");
$hello2 = makeHelloWorld("kevin");
$hello1();
$hello1();
$hello1();
$hello2();
打印結(jié)果:
itbsl0
itbsl1
itbsl2
kevin0
從父作用域繼承變量
在PHP中必須手動調(diào)用閉包對象的bindTo
方法或使用use
關(guān)鍵字把父作用域的變量及狀態(tài)附加到PHP閉包中。而實際應用中,又以使用use
關(guān)鍵字實現(xiàn)居多。
use關(guān)鍵字
實際上,Laravel框架中也大量使用了閉包,最常見的比如路由定義:
Route::group(['domain' => '{account}.myapp.com'], function () {
Route::get('user/{id}', function ($account, $id) {
//
});
});
這里面的兩個function都是匿名函數(shù)。而從父作用域繼承變量的使用場景在Laravel底層源碼中也是俯拾即是,比如Model.php(Illuminate\Database\Eloquent
)的saveOrFail
方法:
該方法的作用是使用事務將模型數(shù)據(jù)保存到數(shù)據(jù)庫,這里面我們使用匿名函數(shù)返回保存狀態(tài),同時使用use關(guān)鍵字將父作用域的$options傳遞給該閉包以便其能夠訪問這個數(shù)據(jù)。
此外,還支持傳遞多個父作用域變量到匿名函數(shù),比如還是在Model類中的forceFill方法:
多個變量以逗號分隔即可。
bindTo方法
我們在前面已經(jīng)提到,閉包是一個對象,所以我們可以在閉包中使用$this關(guān)鍵字獲取閉包的內(nèi)部狀態(tài),閉包對象的默認狀態(tài)沒什么用,需要注意的是其中的__invoke
魔術(shù)方法和bindTo
方法。
__invoke
的作用前面已經(jīng)說過,當嘗試以調(diào)用函數(shù)的方式調(diào)用一個對象時,__invoke()
方法會被自動調(diào)用。
接下來我們來看看bindTo
方法,通過該方法,我們可以把閉包的內(nèi)部狀態(tài)綁定到其他對象上。這里bindTo
方法的第二個參數(shù)顯得尤為重要,其作用是指定綁定閉包的那個對象所屬的PHP類,這樣,閉包就可以在其他地方訪問邦定閉包的對象中受保護和私有的成員變量。
你會發(fā)現(xiàn),PHP框架經(jīng)常使用bindTo
方法把路由URL映射到匿名回調(diào)函數(shù)上,框架會把匿名回調(diào)函數(shù)綁定到應用對象上,這樣在匿名函數(shù)中就可以使用$this
關(guān)鍵字引用重要的應用對象:
class App {
protected $routes = [];
protected $responseStatus = '200 OK';
protected $responseContentType = 'text/html';
protected $responseBody = 'Laravel學院';
public function addRoute($routePath, $routeCallback) {
$this->routes[$routePath] = $routeCallback->bindTo($this, __CLASS__);
}
public function dispatch($currentPath) {
foreach ($this->routes as $routePath => $callback) {
if( $routePath === $currentPath) {
$callback();
}
}
header('HTTP/1.1 ' . $this->responseStatus);
header('Content-Type: ' . $this->responseContentType);
header('Content-Length: ' . mb_strlen($this->responseBody));
echo $this->responseBody;
}
}
這里我們需要重點關(guān)注addRoute方法,這個方法的參數(shù)分別是一個路由路徑和一個路由回調(diào),dispatch方法的參數(shù)是當前HTTP請求的路徑,它會調(diào)用匹配的路由回調(diào)。第9行是重點所在,我們將路由回調(diào)綁定到了當前的App實例上。這么做能夠在回調(diào)函數(shù)中處理App實例的狀態(tài):
$app = new App();
$app->addRoute(‘user/nonfu', function(){
$this->responseContentType = ‘a(chǎn)pplication/json;charset=utf8';
$this->responseBody = ‘{“name”:”LaravelAcademy"}';
});
$app->dispatch(‘user/nonfu');
在Larval底層也有用到bindTo方法,詳見Illuminate\Support\Traits\Macroable的__call方法:
以上所述是小編給大家介紹的PHP匿名函數(shù)和閉包詳解整合,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
您可能感興趣的文章:- 淺析PHP中的閉包和匿名函數(shù)
- php的閉包(Closure)匿名函數(shù)初探
- php的閉包(Closure)匿名函數(shù)詳解
- PHP中的閉包(匿名函數(shù))淺析