以下是【金聰采編】分享的內(nèi)容全文:
概述
閉包和匿名函數(shù)在PHP 5.3.0中引入,這兩個(gè)特性非常有用,每個(gè)PHP開(kāi)發(fā)者都應(yīng)該掌握。
匿名函數(shù)其實(shí)就是沒(méi)有名稱(chēng)的函數(shù),匿名函數(shù)可以賦值給變量,還能像其他任何PHP函數(shù)對(duì)象那樣傳遞。不過(guò)匿名函數(shù)仍然是函數(shù),因此可以調(diào)用,還可以傳入?yún)?shù),適合作為函數(shù)或方法的回調(diào)。
閉包是指在創(chuàng)建時(shí)封裝周?chē)鸂顟B(tài)的函數(shù),即使閉包所在的環(huán)境的不存在了,閉包中封裝的狀態(tài)依然存在。
創(chuàng)建匿名函數(shù)
創(chuàng)建匿名函數(shù)很簡(jiǎn)單:
//將匿名函數(shù)賦給一個(gè)變量,通過(guò)變量名+()的形式來(lái)調(diào)用$greet = function () { return "Hello World";};echo $greet();結(jié)果打印:
Hello World
匿名函數(shù)和普通的PHP函數(shù)很像:常用的句法相同,也接受參數(shù),而且能返回值。不過(guò)閉包沒(méi)有函數(shù)名。
注:我們之所以能調(diào)用$greet變量,是因?yàn)檫@個(gè)變量的值是一個(gè)閉包,而且閉包對(duì)象實(shí)現(xiàn)了__invoke()魔術(shù)方法,只要變量名后有(),PHP就會(huì)查找并調(diào)用__invoke方法。
我們通常把匿名函數(shù)當(dāng)做函數(shù)或方法的回調(diào)使用,事實(shí)上,很多PHP函數(shù)都會(huì)用到匿名函數(shù),比如array_map和preg_replace_callback,這是使用PHP匿名函數(shù)的絕佳時(shí)機(jī)。記住,閉包和其他值一樣,可以作為參數(shù)傳入其他PHP函數(shù):
$numberPlusOne = array_map(function ($number) { return $number += 1;}, [1, 2, 3]);print_r($numberPlusOne);在匿名函數(shù)出現(xiàn)之前,要實(shí)現(xiàn)這樣的功能,PHP開(kāi)發(fā)者只能單獨(dú)創(chuàng)建具名函數(shù),然后使用名稱(chēng)引用這個(gè)函數(shù):
function incrementNumber ($number) { return $number += 1;}$numberPlusOne = array_map(‘incrementNumber', [1, 2, 3]);print_r($numberPlusOne);這樣做把回調(diào)的實(shí)現(xiàn)和使用場(chǎng)所隔離開(kāi)了,而且使用閉包實(shí)現(xiàn)代碼更加簡(jiǎn)潔。
創(chuàng)建閉包
包含自由變量的函數(shù)與為所有這些自由變量提供了變量綁定的環(huán)境一起,被稱(chēng)為閉包。
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中必須手動(dòng)調(diào)用閉包對(duì)象的bindTo方法或使用use關(guān)鍵字把父作用域的變量及狀態(tài)附加到PHP閉包中。而實(shí)際應(yīng)用中,又以使用use關(guān)鍵字實(shí)現(xiàn)居多。
use關(guān)鍵字
實(shí)際上,Laravel框架中也大量使用了閉包,最常見(jiàn)的比如路由定義:
Route::group(['domain' => '{account}.myapp.com'], function () { Route::get('user/{id}', function ($account, $id) { // });});這里面的兩個(gè)function都是匿名函數(shù)。而從父作用域繼承變量的使用場(chǎng)景在Laravel底層源碼中也是俯拾即是,比如Model.php(Illuminate/Database/Eloquent)的saveOrFail方法:
![]() |
該方法的作用是使用事務(wù)將模型數(shù)據(jù)保存到數(shù)據(jù)庫(kù),這里面我們使用匿名函數(shù)返回保存狀態(tài),同時(shí)使用use關(guān)鍵字將父作用域的$options傳遞給該閉包以便其能夠訪問(wèn)這個(gè)數(shù)據(jù)。
此外,還支持傳遞多個(gè)父作用域變量到匿名函數(shù),比如還是在Model類(lèi)中的forceFill方法:
![]() |
多個(gè)變量以逗號(hào)分隔即可。
bindTo方法
我們?cè)谇懊嬉呀?jīng)提到,閉包是一個(gè)對(duì)象,所以我們可以在閉包中使用$this關(guān)鍵字獲取閉包的內(nèi)部狀態(tài),閉包對(duì)象的默認(rèn)狀態(tài)沒(méi)什么用,需要注意的是其中的__invoke魔術(shù)方法和bindTo方法。
__invoke的作用前面已經(jīng)說(shuō)過(guò),當(dāng)嘗試以調(diào)用函數(shù)的方式調(diào)用一個(gè)對(duì)象時(shí),__invoke() 方法會(huì)被自動(dòng)調(diào)用。
接下來(lái)我們來(lái)看看bindTo方法,通過(guò)該方法,我們可以把閉包的內(nèi)部狀態(tài)綁定到其他對(duì)象上。這里bindTo方法的第二個(gè)參數(shù)顯得尤為重要,其作用是指定綁定閉包的那個(gè)對(duì)象所屬的PHP類(lèi),這樣,閉包就可以在其他地方訪問(wèn)邦定閉包的對(duì)象中受保護(hù)和私有的成員變量。
你會(huì)發(fā)現(xiàn),PHP框架經(jīng)常使用bindTo方法把路由URL映射到匿名回調(diào)函數(shù)上,框架會(huì)把匿名回調(diào)函數(shù)綁定到應(yīng)用對(duì)象上,這樣在匿名函數(shù)中就可以使用$this關(guān)鍵字引用重要的應(yīng)用對(duì)象:
class App { protected $routes = []; protected $responseStatus = '200 OK'; protected $responseContentType = 'text/html'; protected $responseBody = 'Laravel學(xué)院'; 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; }}這里我們需要重點(diǎn)關(guān)注addRoute方法,這個(gè)方法的參數(shù)分別是一個(gè)路由路徑和一個(gè)路由回調(diào),dispatch方法的參數(shù)是當(dāng)前HTTP請(qǐng)求的路徑,它會(huì)調(diào)用匹配的路由回調(diào)。第9行是重點(diǎn)所在,我們將路由回調(diào)綁定到了當(dāng)前的App實(shí)例上。這么做能夠在回調(diào)函數(shù)中處理App實(shí)例的狀態(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方法,詳見(jiàn)Illuminate/Support/Traits/Macroable的__call方法:以上所述是小編給大家介紹的PHP匿名函數(shù)和閉包詳解整合,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)金聰精品網(wǎng)站的支持!
1.軟件源碼推廣展示:目的展示軟件相關(guān)功能,接收技術(shù)學(xué)習(xí)者測(cè)試、測(cè)評(píng);
2.教程課程信息展示:展示課程信息,傳授課程各階段內(nèi)容;
3.設(shè)計(jì)素材圖片展示:展示素材設(shè)計(jì)理念、思維方式、傳播設(shè)計(jì)理念;
4.福利優(yōu)惠信息展示:分享各類(lèi)最新的福利信息,各種優(yōu)惠信息展示;
以上分享目的僅供學(xué)習(xí)、參考使用,請(qǐng)勿用于其他用途,如果想商業(yè)使用或者代理,請(qǐng)自行聯(lián)系版權(quán)方獲取授權(quán)。任何未獲取授權(quán)的商業(yè)使用與本站無(wú)關(guān),請(qǐng)自行承擔(dān)相應(yīng)責(zé)任。
本站不存儲(chǔ)任何資源文件,敬請(qǐng)周知!
如果您認(rèn)為本頁(yè)信息內(nèi)容侵犯了您的相關(guān)權(quán)益(包含但不限于:著作權(quán)、首發(fā)權(quán)、隱私權(quán)等權(quán)利),或者您認(rèn)為自己是此信息的權(quán)利人但是此信息不是自己發(fā)布的,可以直接版權(quán)舉報(bào)投訴,我們會(huì)根據(jù)網(wǎng)站注冊(cè)協(xié)議、資源分享協(xié)議等協(xié)議處理,以保護(hù)您的合法權(quán)益。
本網(wǎng)站采用 BY-NC-SA 協(xié)議進(jìn)行授權(quán) 轉(zhuǎn)載請(qǐng)注明原文鏈接:淺談PHP匿名函數(shù)和閉包

侵權(quán)舉報(bào)/版權(quán)申訴





