最经常用作回调函数(callback)参数的值 。当然,也有其它应用的情况。
Example #1 匿名函数示例
(也就是以前传函数名,现在可以直接传匿名函数作为参数了)
<?php
echo preg_replace_callback ( '/-([a-z])/' , function ( $match ) {
return strtoupper ( $match [ 1 ]);
}, 'hello-world' );
// 输出 helloWorld
?>
闭包函数也可以作为变量的值来使用。PHP 会自动把此种表达式转换成内置类 Closure 的对象实例。
把一个 closure 对象赋值给一个变量的方式与普通变量赋值的语法是一样的,最后也要加上分号:
Example #2 匿名函数变量赋值示例
<?php
$greet = function( $name )
{
printf ( "Hello %s\r\n" , $name );
};
$greet ( 'World' );
$greet ( 'PHP' );
?>
Closure 对象也会从父作用域中继承类属性。这些变量都必须在函数或类的头部声明。从父作用域中继承变量与使用全局变量是不同的。全局变量存在于一个全局的范围,无论当前在执行的是哪个函数。而 closure 的父作用域则是声明该 closure 的函数(不一定要是它被调用的函数)。示例如下:
Example #3 Closures 和作用域
<?php
// 一个基本的购物车,包括一些已经添加的商品和每种商品的数量。
// 其中有一个方法用来计算购物车中所有商品的总价格,该方法使
// 用了一个 closure 作为回调函数。
class Cart {
const PRICE_BUTTER = 1.00;
const PRICE_MILK = 3.00;
const PRICE_EGGS = 6.95;
protected $products = array();
public function add($product, $quantity) {
$this->products[$product] = $quantity;
}
public function getQuantity($product) {
return isset($this->products[$product]) ? $this->products[$product] :
FALSE;
}
public function getTotal($tax) {
$total = 0.00;
$callback = function ($quantity, $product) use ($tax, &$total) {
$pricePerItem = constant(__CLASS__ . "::PRICE_" . strtoupper($product));
$total += ($pricePerItem * $quantity) * ($tax + 1.0);
};
array_walk($this->products, $callback);
return round($total, 2);
}
}
$my_cart = new Cart;
// 往购物车里添加条目
$my_cart->add('butter', 1);
$my_cart->add('milk', 3);
$my_cart->add('eggs', 6);
// 打出出总价格,其中有 5% 的销售税.
print $my_cart->getTotal(0.05) . "\n";
// 最后结果是 54.29
?>
这里要注意理解当使用 array_walk($this->products, $callback) 调用 function($quantity, $product) use ($tax, &$total).
这里 $quantity, $product 分别是 $this->products 的key和value,属于匿名函数接收到参数。
而 use 后面跟的是匿名函数外部作用域的变量。
更新日志
版本 说明
5.4.0 $this 可用于匿名函数。
5.3.0 可以使用匿名函数。
注释
Note: 可以在 closure 中使用 func_num_args() , func_get_arg() 和 func_get_args() 。