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

蕭瀟發表於2019-03-04

閉包

閉包是什麼?

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語言特點的深入研究。

相關文章