PHP新特性之閉包、匿名函數
閉包
閉包是什麼?
1).閉包和匿名函數在PHP5.3中被引入。
2).閉包是指在創建時封裝函數周圍狀態的函數,即使閉包所在的環境不存在了,閉包封裝的狀態依然存在,這一點和Javascript的閉包特性很相似。3).匿名函數就是沒有名稱的函數,匿名函數可以賦值給變數,還可以像其他任何PHP對象一樣傳遞。可以將匿名函數和閉包視作相同的概念。
4).需要注意的是閉包使用的語法和普通函數相同,但是他其實是偽裝成函數的對象,是Closure類的實例。閉包和字元串或整數一樣,是一等值類型。創建閉包
<?phpn$closure = function () {n return sprintf(hello %s, $name);n}nnecho $closure(Josh);n
我們之所以可以調用$closure變數,是因為這個變數的值是一個閉包,而且閉包對象實現了__invoke()魔術方法,只要後面跟著(),PHP就會查找__invoke()方法。這裡簡單解釋下這個魔術方法:
<?phpnclass testClassn{n public function __invoken {n print "hello world";n }n}n$n = new testClass;n$n();n
PHP自從5.3版以來就新增了一個叫做__invoke()的魔術方法,使用該方法就可以在創建實例後,直接調用對象。
何時使用?
我們通常把PHP閉包當做函數和方法的回調使用。很多PHP函數都會用到回調函數,例如array_map()和preg_replace_callback()。
<?phpn$numbersPlusOne = array_map(function($number) {n return $number + 1;n}, [1, 2, 3]);n
如何理解附加狀態?
1).注意PHP閉包不會真的像JS一樣自動封裝應用的狀態,在PHP中必須調用閉包對象的bindTo方法或者使用use關鍵字,把狀態附加到PHP閉包上。來看一個例子
<?phpnfunction enclosePerson($name)n{n return function ($doCommand) use ($name) {n return sprintf(%s , %s, $name, $doCommand);n } n}n//把字元串「Clay」封裝在閉包中n$clay = enclosePerson(Clay);n//傳入參數,調用閉包necho $clay(get me sweat tea!); // Clay, get me sweat tea!n
在這個例子中,函數enclosePerson()有一個$name參數,這個函數返回一個閉包對象,這個閉包封裝了$name參數,即便返回的對象跳出了enclosePerson()函數的作用域,它也會記住$name參數的值,因為$name變數仍然在閉包中。
2).使用use關鍵字可以把多個關鍵字傳入閉包,此時要想像PHP函數或方法的參數一樣,使用逗號分割多個參數。3).PHP閉包仍然是對象,可以使用$this關鍵字獲取閉包的內部狀態。閉包的默認狀態裡面有一個__invoke()魔術方法和bindTo()方法。4).bindTo()方法為閉包增加了一些有趣的東西。我們可以使用這個方法把Closure對象內部狀態綁定到其他對象上。bindTo()方法的第二個參數可以指定綁定閉包的那個對象所屬的PHP類,這樣我們就可以訪問這個類的受保護和私有的成員變數。看下面的代碼示例:<?phpnclass Appn{n protected $route = array();n protected $responseStatus = 200 OK;n protected $responseContentType = text/html;n protected $responseBody = Hello world;nn public function addRoute($routePath, $routeCallback)n {n $this->routes[$routePath] = $routeCallback->bindTo($this, __CLASS__);n }nn public function dispatch($currentPath)n {n foreach($this->routes as $routePath => $callback) {n if ($routePath === $currentPath) {n $callback();n }n }n header(HTTP/1.1 . $this->responseStatus);n header(Content-type: . $this->responseContentType);n header(Content-length: . mb_strlen($this->responseBody));n echo $this->responseBody;n }n}n
我們把路由回調綁定到了當前的App實例上,這樣就可以在回調函數中處理App實例的狀態了。
<?phpn$app = new App();n$app->addRoute(/users/xiaoxiao, function () {n $this->responseContentType = application/json;charset=utf8;n $this->responseBody = {"name" : "xiaoxiao"};n});n$app->dispatch(/users/xiaoxiao);n
推薦閱讀: