1、概述
閉包和匿名函數在PHP 5.3.0中引入,這兩個特性非常有用,每個PHP開發者都應該掌握。
閉包是指在創建時封裝周圍狀態的函數,即使閉包所在的環境的不存在了,閉包中封裝的狀態依然存在。
匿名函數其實就是沒有名稱的函數,匿名函數可以賦值給變量,還能像其他任何PHP函數對象那樣傳遞。不過匿名函數仍然是函數,因此可以調用,還可以傳入參數,適合作為函數或方法的回調。
注:理論上講閉包和匿名函數是不同的概念,不過PHP將其視作相同的概念(匿名函數在PHP中也叫作閉包函數),所以下面提到閉包時指的也是匿名函數;反之亦然。
2、創建閉包
創建閉包很簡單:
- $greet = function ($name) {
- return sprintf("Hello %s/r/n", $name);
- };
- echo $greet('Vevb.com');
結果打印:
Hello Vevb.com
閉包和普通的PHP函數很像:常用的句法相同,也接受參數,而且能返回值。不過閉包沒有函數名。
注:我們之所以能調用$greet變量,是因為這個變量的值是一個閉包,而且閉包對象實現了__invoke()魔術方法,只要變量名后有(),PHP就會查找并調用__invoke方法。
我們通常把PHP閉包當做函數會方法的回調使用,事實上,很多PHP函數都會用到閉包,比如array_map和preg_replace_callback,這是使用PHP匿名函數的絕佳時機。記住,閉包和其他值一樣,可以作為參數傳入其他PHP函數:
- $numberPlusOne = array_map(function ($number) {
- return $number += 1;
- }, [1, 2, 3]);
- print_r($numberPlusOne);
在閉包出現之前,要實現這樣的功能,PHP開發者只能單獨創建具名函數,然后使用名稱引用這個函數:
- function incrementNumner ($number) {
- return $number += 1;
- }
- $numberPlusOne = array_map(‘incrementNumber’, [1, 2, 3]);
- print_r($numberPlusOne);
這樣做把回調的實現和使用場所隔離開了,而且使用閉包實現代碼更加簡潔。
3、從父作用域繼承變量
在PHP中必須手動調用閉包對象的bindTo方法或使用use關鍵字把父作用域的變量及狀態附加到PHP閉包中。而實際應用中,又以使用use關鍵字實現居多。
use關鍵字
實際上,Laravel框架中也大量使用了閉包,最常見的比如路由定義:
- Route::group(['domain' => '{account}.myapp.com'], function () {
- Route::get('user/{id}', function ($account, $id) {
- //
- });
- });
這里面的兩個function都是閉包。而從父作用域繼承變量的使用場景在Laravel底層源碼中也是俯拾即是,比如Model.php(Illuminate/Database/Eloquent)的saveOrFail方法:
closure-use
該方法的作用是使用事務將模型數據保存到數據庫,這里面我們使用閉包返回保存狀態,同時使用use關鍵字將父作用域的$options傳遞給該閉包以便其能夠訪問這個數據。
此外,還支持傳遞多個父作用域變量到閉包,比如還是在Model類中的forceFill方法:
closure-use-multi
多個變量以逗號分隔即可。
bindTo方法
我們在前面已經提到,閉包是一個對象,所以我們可以在閉包中使用$this關鍵字獲取閉包的內部狀態,閉包對象的默認狀態沒什么用,需要注意的是其中的__invoke魔術方法和bindTo方法。
__invoke的作用前面已經說過,當嘗試以調用函數的方式調用一個對象時,__invoke() 方法會被自動調用。
接下來我們來看看bindTo方法,通過該方法,我們可以把閉包的內部狀態綁定到其他對象上。這里bindTo方法的第二個參數顯得尤為重要,其作用是指定綁定閉包的那個對象所屬的PHP類,這樣,閉包就可以在其他地方訪問邦定閉包的對象中受保護和私有的成員變量。
你會發現,PHP框架經常使用bindTo方法把路由URL映射到匿名回調函數上,框架會把匿名回調函數綁定到應用對象上,這樣在匿名函數中就可以使用$this關鍵字引用重要的應用對象:
- 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();
- } //Vevb.com
- }
- header('HTTP/1.1 ' . $this->responseStatus);
- header('Content-Type: ' . $this->responseContentType);
- header('Content-Length: ' . mb_strlen($this->responseBody));
- echo $this->responseBody;
- }
- }
這里我們需要重點關注addRoute方法,這個方法的參數分別是一個路由路徑和一個路由回調,dispatch方法的參數是當前HTTP請求的路徑,它會調用匹配的路由回調。第9行是重點所在,我們將路由回調綁定到了當前的App實例上。這么做能夠在回調函數中處理App實例的狀態:
- $app = new App();
- $app->addRoute(‘user/nonfu’, function(){
- $this->responseContentType = ‘application/json;charset=utf8’;
- $this->responseBody = ‘{“name”:”LaravelAcademy"}';
- });
- $app->dispatch(‘user/nonfu');
在Larval底層也有用到bindTo方法,詳見Illuminate/Support/Traits/Macroable的__call方法。
新聞熱點
疑難解答