匿名函数就是没有名称的函数。匿名函数可以赋值给变量,还能像其他任何PHP对象那样传递。不过匿名函数仍是函数,因此可以调用,还可以传入参数。匿名函数特别适合作为函数或方法的回调。
注意:理论上讲,闭包和匿名函数是不同的概念。不过,PHP将其视作相同的概念。所以,我们提到闭包时,指的也是匿名函数,反之亦然。
PHP闭包和匿名函数使用的句法与普通函数相同,但闭包和匿名函数其实是 伪装成函数的对象(Closure类的实例) 。
创建闭包
1
2
3
4
5
6
|
$closure
=
function
(
$name
){
return
sprintf(
"Hello %s"
,
$name
);
}
echo
$closure
(
"jerry"
);
// 检测$closure变量是否是一个闭包
var_dump(
$closure
instanceof
Closure);
|
以上代码创建了一个闭包对象,然后将其赋值给 $closure 变量。闭包和普通的PHP函数很像,使用的句法相同,也接收参数,而且能返回值。
说明:我们之所以能够调用 $closure 变量,是因为这个变量的值是一个闭包,而且闭包对象实现了 __invoke() 魔术方法。只要变量名后有 () ,PHP就会查找并调用 __invoke()
方法。
使用闭包
我们通常把PHP闭包当做当做函数和方法的回调使用。很多PHP函数都会用到回调函数,例如 array_map()
和 preg_replace_callback()
.如下示例,我们将用 array_map() 处理数组,将数组每一项自增1:
1
2
3
4
|
$nubmers
=
array_map
(
function
(
$number
){
return
$number
++;
}, [1,2,3]);
var_dump(
$numbers
);
|
附加状态
PHP闭包不会像真正的javascrypt闭包那样自动封装应用的状态,我们必须手动调用闭包对象的 bindTo() 方法或者使用 use 关键字,把状态附加到PHP闭包上。
使用 use 关键字
使用 use 关键字来附加闭包状态更加常见,因此我们先来看这种方式。使用 use 关键字把变量附加闭包上时,附加的变量会记住附加时赋给它的值。
1
2
3
4
5
6
7
8
9
10
|
function
Car (
$name
){
return
function
(
$statu
)
use
(
$name
){
return
sprintf(
"Car %s is %s"
,
$name
,
$statu
);
}
}
// 将车名封装在闭包中
$car
= Car(
"bmw"
);
// 调用车的动作
// 输出--> "bmw is running"
echo
$car
(
"running"
);
|
注意:使用 use 关键字可以把多个参数传入闭包,此时要像PHP函数或方法的参数一样,使用逗号分隔多个参数。
使用 bindTo() 方法附加闭包的状态
与其它PHP对象类似,每个闭包实例都可以使用 $this 关键字获取闭包的内部状态。闭包对象的默认状态没什么用,不过有一个 __invoke() 魔术方法和 bindTo() 方法。
bindTo() 方法为闭包增加了一些有趣的潜力。我们可以使用这个方法把 Closure 对象的内部状态绑定到其它对象上。
bindTo() 方法的第二个参数很重要,其作用是指定绑定闭包的那个对象所属的PHP类。因此,闭包可以访问绑定闭包的对象中受保护和私有的成员变量。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
|
class
TestClosure
{
private
$name
=[];
private
$age
;
private
$sex
;
public
function
addPerson(
$name
,
$personCallback
){
// 将闭包对象绑定当前实例
$this
->name[
$name
] =
$personCallback
->bindTo(
$this
,
__CLASS__
);
}
public
function
display(
$name
){
foreach
(
$this
->name
as
$key
=>
$callback
){
if
(
$key
==
$name
){
// 执行闭包对象,将闭包状态附加到类
$callback
();
}
}
echo
"name : {$name}\n"
;
echo
"age : {$this->age}\n"
;
echo
"sex : {$this->sex}\n"
;
}
}
$person
=
new
TestClosure();
$person
->addPerson(
"jerry"
,
function
(){
$this
->age = 19;
$this
->sex =
"man"
;
});
$person
->display(
"jerry"
);
/** output
name : jerry
age : 19
sex : man
*/
|