PHP新特性之閉包、匿名函數

繼續世界上最好的語言學習,最近一周都是PHP哈哈。下周開始學JS。

閉包

閉包是什麼?

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

推薦閱讀:

php curl抓取不到頁面及來路問題?
如何用php 編寫網路爬蟲?

TAG:匿名函数 | 闭包 | PHP |