PHP閉包函式用法
阿新 • • 發佈:2021-01-26
PHP閉包函式用法
一、閉包
-
1).閉包和匿名函式在PHP5.3中被引入。
-
2).閉包是指在建立時封裝函數週圍狀態的函式,即使閉包所在的環境不存在了,閉包封裝的狀態依然存在,這一點和Javascript的閉包特性很相似。
-
3).匿名函式就是沒有名稱的函式,匿名函式可以賦值給變數,還可以像其他任何PHP物件一樣傳遞。可以將匿名函式和閉包視作相同的概念。
-
4).需要注意的是閉包使用的語法和普通函式相同,但是他其實是偽裝成函式的物件,是Closure類的例項。閉包和字串或整數一樣,是一等值型別。
二、閉包使用
1,匿名函式
$變數名=functio(){ };
2,作為回撥函式
1)array_filter($陣列,function( $v){ 規則 }); 根據規則篩選陣列內容
2)$numbersPlusOne = array_map(function($number) {
return $number + 1;
}, [1, 2, 3]);
根據函式內容,作用到陣列每個值中,並且返回新陣列
三,附加狀態
1).注意PHP閉包不會真的像JS一樣自動封裝應用的狀態,在PHP中必須呼叫閉包物件的bindTo方法或者使用use關鍵字,把狀態附加到PHP閉包上。
<?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類,這樣我們就可以訪問這個類的受保護和私有的成員變數。
<?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例項的狀態了。
<?php
$app = new App();
$app->addRoute('/users/xiaoxiao', function () {
$this->responseContentType = 'application/json;charset=utf8';
$this->responseBody = '{"name" : "xiaoxiao"}';
});
$app->dispatch('/users/xiaoxiao');