匿名函數php
匿名函數,也叫閉包函數,說白了就是「沒有名字的函數」,和通常函數結構同樣,只是少了函數名以及最後須要加上分號;
。html
注:理論上講閉包和匿名函數是不一樣的概念,不過PHP將其視做相同的概念。
$func = function()
{
echo 'Hello World' . PHP_EOL;
};
$func();複製代碼
匿名函數和普通函數的區分有:laravel
$message = 'hello';
$example = function () use ($message) {
return $message;
};
$message = 'world';
echo $example();
輸出:hello複製代碼
注意:必須使用use
關鍵字將變量傳遞進去才行,具體見官方文檔。面試
定義一個閉包函數,其實就是實例化一個閉包類(Closure
)對象:sql
$func = function()
{
echo 'hello world' . PHP_EOL;
};
var_dump($func);
輸出:
object(Closure)#1 (0) {
}複製代碼
類摘要:shell
Closure {
__construct ( void )
public static Closure bind ( Closure $closure , object $newthis [, mixed $newscope = 'static' ] )
public Closure bindTo ( object $newthis [, mixed $newscope = 'static' ] )
}複製代碼
除了以上方法,閉包還實現了一個__invoke()
魔術方法,當嘗試以調用函數的方式調用一個對象時,__invoke()
方法會被自動調用。json
接下來咱們來看看bindTo
方法,經過該方法,咱們能夠把閉包的內部狀態綁定到其餘對象上。這裏bindTo
方法的第二個參數顯得尤其重要,其做用是指定綁定閉包的那個對象所屬的PHP類,這樣,閉包就能夠在其餘地方訪問綁定閉包的對象中受保護和私有的成員變量。bash
你會發現,PHP框架常常使用bindTo
方法把路由URL映射到匿名回調函數上,框架會把匿名回調函數綁定到應用對象上,這樣在匿名函數中就可使用$this
關鍵字引用重要的應用對象:服務器
class App {
protected $routes = [];
protected $responseStatus = '200 OK';
protected $responseContentType = 'text/html';
protected $responseBody = 'Hello World';
public function addRoute($path, $callback) {
$this->routes[$path] = $callback->bindTo($this, __CLASS__);
}
public function dispatch($path) {
foreach ($this->routes as $routePath => $callback) {
if( $routePath === $path) {
$callback();
}
}
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’, function(){
$this->responseContentType = ‘application/json;charset=utf8’;
$this->responseBody = '世界你好';
});
$app->dispatch('/user');複製代碼
匿名函數能夠從父做用域繼承變量,而這個父做用域是定義該閉包的函數(不必定是調用它的函數)。
利用這個特性,咱們能夠實現一個簡單的控制反轉IoC
容器:
class Container
{
protected static $bindings;
public static function bind($abstract, Closure $concrete)
{
static::$bindings[$abstract] = $concrete;
}
public static function make($abstract)
{
return call_user_func(static::$bindings[$abstract]);
}
}
class talk
{
public function greet($target)
{
echo 'Hello ' . $target->getName();
}
}
class A
{
public function getName()
{
return 'World';
}
}
// 建立一個talk類的實例
$talk = new talk();
// 將A類綁定至容器,命名爲foo
Container::bind('foo', function() {
return new A;
});
// 經過容器取出實例
$talk->greet(Container::make('foo')); // Hello World複製代碼
上述例子中,只有在經過make
方法獲取實例的時候,實例才被建立,這樣使得咱們能夠實現容器。
在Laravel
框架底層也大量使用了閉包以及bindTo
方法,利用好閉包能夠實現更多的高級特性如事件觸發等。
以上內容但願幫助到你們, 不少PHPer在進階的時候總會遇到一些問題和瓶頸,業務代碼寫多了沒有方向感,不知道該從那裏入手去提高,對此我整理了一些資料,包括但不限於:分佈式架構、高可擴展、高性能、高併發、服務器性能調優、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql優化、shell腳本、Docker、微服務、Nginx等多個知識點高級進階乾貨須要的能夠免費分享給你們 ,須要戳這裏 PHP進階架構師>>>視頻、面試文檔免費獲取