php _invoke 闭包,PHP新特性之闭包、匿名函数
閉包
閉包是什么?
1).閉包和匿名函數在PHP5.3中被引入。
2).閉包是指在創建時封裝函數周圍狀態的函數,即使閉包所在的環境不存在了,閉包封裝的狀態依然存在,這一點和Javascript的閉包特性很相似。
3).匿名函數就是沒有名稱的函數,匿名函數可以賦值給變量,還可以像其他任何PHP對象一樣傳遞。可以將匿名函數和閉包視作相同的概念。
4).需要注意的是閉包使用的語法和普通函數相同,但是他其實是偽裝成函數的對象,是Closure類的實例。閉包和字符串或整數一樣,是一等值類型。
創建閉包
$closure = function ($name){
return sprintf('hello %s', $name);
};
echo $closure('Josh');復制代碼
我們之所以可以調用$closure變量,是因為這個變量的值是一個閉包,而且閉包對象實現了__invoke()魔術方法,只要后面跟著(),PHP就會查找__invoke()方法。這里簡單解釋下這個魔術方法:
class testClass{
public function __invoke{
print "hello world";
}
}
$n = new testClass;
$n();復制代碼
PHP自從5.3版以來就新增了一個叫做__invoke()的魔術方法,使用該方法就可以在創建實例后,直接調用對象。
何時使用?
我們通常把PHP閉包當做函數和方法的回調使用。很多PHP函數都會用到回調函數,例如array_map()和preg_replace_callback()。
$numbersPlusOne = array_map(function($number){
return $number + 1;
}, [1, 2, 3]);復制代碼
如何理解附加狀態?
1).注意PHP閉包不會真的像JS一樣自動封裝應用的狀態,在PHP中必須調用閉包對象的bindTo方法或者使用use關鍵字,把狀態附加到PHP閉包上。來看一個例子
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類,這樣我們就可以訪問這個類的受保護和私有的成員變量。看下面的代碼示例:
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專題系列目錄地址:github.com/xx19941215/…
PHP專題系列預計寫二十篇左右,主要總結我們日常PHP開發中容易忽略的基礎知識和現代PHP開發中關于規范、部署、優化的一些實戰性建議,同時還有對Javascript語言特點的深入研究。
總結
以上是生活随笔為你收集整理的php _invoke 闭包,PHP新特性之闭包、匿名函数的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 建行信用卡网上申请要多久 网上申请既方便
- 下一篇: 动态规划算法php,php算法学习之动态