1).閉包和匿名函數在PHP5.3中被引入。
2).閉包是指在建立時封裝函數週圍狀態的函數,即便閉包所在的環境不存在了,閉包封裝的狀態依然存在,這一點和Javascript的閉包特性很類似。
3).匿名函數就是沒有名稱的函數,匿名函數能夠賦值給變量,還能夠像其餘任何PHP對象同樣傳遞。能夠將匿名函數和閉包視做相同的概念。
4).須要注意的是閉包使用的語法和普通函數相同,可是他實際上是假裝成函數的對象,是Closure類的實例。閉包和字符串或整數同樣,是一等值類型。php
$closure = function ($name) { return sprintf('hello %s', $name); }; echo $closure('Josh');
咱們之因此能夠調用$closure變量,是由於這個變量的值是一個閉包,並且閉包對象實現了__invoke()
魔術方法,只要後面跟着()
,PHP就會查找__invoke()
方法。這裏簡單解釋下這個魔術方法:html
class testClass { public function __invoke { print "hello world"; } } $n = new testClass; $n();
PHP自從5.3版以來就新增了一個叫作__invoke()
的魔術方法,使用該方法就能夠在建立實例後,直接調用對象。git
咱們一般把PHP閉包當作函數和方法的回調使用。不少PHP函數都會用到回調函數,例如array_map()
和preg_replace_callback()
。github
$numbersPlusOne = array_map(function($number) { return $number + 1; }, [1, 2, 3]);
1).注意PHP閉包不會真的像JS同樣自動封裝應用的狀態,在PHP中必須調用閉包對象的bindTo方法或者使用use關鍵字,把狀態附加到PHP閉包上。來看一個例子web
function enclosePerson($name) { return function ($doCommand) use ($name) { return sprintf('%s , %s', $name, $doCommand); }; } //把字符串「Clay」封裝在閉包中 $clay = enclosePerson('Clay'); //傳入參數,調用閉包 echo $clay('get me sweat tea!'); // Clay, get me sweat tea!
在這個例子中,函數enclosePerson()
有一個$name
參數,這個函數返回一個閉包對象,這個閉包封裝了$name
參數,即使返回的對象跳出了enclosePerson()
函數的做用域,它也會記住$name
參數的值,由於$name
變量仍然在閉包中。
2).使用use關鍵字能夠把多個關鍵字傳入閉包,此時要想像PHP函數或方法的參數同樣,使用逗號分割多個參數。
3).PHP閉包仍然是對象,可使用$this關鍵字獲取閉包的內部狀態。閉包的默認狀態裏面有一個__invoke()
魔術方法和bindTo()
方法。
4).bindTo()
方法爲閉包增長了一些有趣的東西。咱們可使用這個方法把Closure對象內部狀態綁定到其餘對象上。bindTo()
方法的第二個參數能夠指定綁定閉包的那個對象所屬的PHP類,這樣咱們就能夠訪問這個類的受保護和私有的成員變量。看下面的代碼示例:json
class App { protected $route = array(); protected $responseStatus = '200 OK'; protected $responseContentType = 'text/html'; protected $responseBody = 'Hello world'; 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; } }
咱們把路由回調綁定到了當前的App實例上,這樣就能夠在回調函數中處理App實例的狀態了。閉包
$app = new App(); $app->addRoute('/users/xiaoxiao', function () { $this->responseContentType = 'application/json;charset=utf8'; $this->responseBody = '{"name" : "xiaoxiao"}'; }); $app->dispatch('/users/xiaoxiao');
PHP專題系列目錄地址:https://github.com/xx19941215/webBlog
PHP專題系列預計寫二十篇左右,主要總結咱們平常PHP開發中容易忽略的基礎知識和現代PHP開發中關於規範、部署、優化的一些實戰性建議,同時還有對Javascript語言特色的深刻研究。app