PHP 閉包那些事兒

匿名函數

匿名函數,也叫閉包函數,說白了就是「沒有名字的函數」,和通常函數結構同樣,只是少了函數名以及最後須要加上分號;php

注:理論上講閉包和匿名函數是不一樣的概念,不過PHP將其視做相同的概念。html

$func = function()
{
    echo 'Hello World' . PHP_EOL;
};
$func();
複製代碼

匿名函數和普通函數的區分有:json

  • 匿名函數也能夠做爲變量的值來使用。
  • 匿名函數能夠從父做用域繼承變量,而這個父做用域是定義該閉包的函數(不必定是調用它的函數)。
$message = 'hello';
$example = function () use ($message) {
    return $message;
};
$message = 'world';
echo $example();

輸出:hello
複製代碼

注意:必須使用use關鍵字將變量傳遞進去才行,具體見官方文檔bash

閉包類

定義一個閉包函數,其實就是實例化一個閉包類(Closure)對象:閉包

$func = function()
{
    echo 'hello world' . PHP_EOL;
};
var_dump($func);

輸出:
object(Closure)#1 (0) {
}
複製代碼

類摘要:app

Closure {
     __construct ( void )
     public static Closure bind ( Closure $closure , object $newthis [, mixed $newscope = 'static' ] )
     public Closure bindTo ( object $newthis [, mixed $newscope = 'static' ] )
}
複製代碼

除了以上方法,閉包還實現了一個__invoke()魔術方法,當嘗試以調用函數的方式調用一個對象時,__invoke()方法會被自動調用。框架

bindTo 方法

接下來咱們來看看bindTo方法,經過該方法,咱們能夠把閉包的內部狀態綁定到其餘對象上。這裏bindTo方法的第二個參數顯得尤其重要,其做用是指定綁定閉包的那個對象所屬的PHP類,這樣,閉包就能夠在其餘地方訪問綁定閉包的對象中受保護和私有的成員變量。函數

你會發現,PHP框架常用bindTo方法把路由URL映射到匿名回調函數上,框架會把匿名回調函數綁定到應用對象上,這樣在匿名函數中就能夠使用$this關鍵字引用重要的應用對象:ui

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實例的狀態:this

$app = new App();
$app->addRoute(‘/user’, function(){
    $this->responseContentType = ‘application/json;charset=utf8’;
    $this->responseBody = '世界你好';
});
$app->dispatch('/user');
複製代碼

IoC 容器

匿名函數能夠從父做用域繼承變量,而這個父做用域是定義該閉包的函數(不必定是調用它的函數)。

利用這個特性,咱們能夠實現一個簡單的控制反轉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方法,利用好閉包能夠實現更多的高級特性如事件觸發等。

相關文章
相關標籤/搜索