匿名函数小编使用得非常的少了在开发时一般用不到php匿名函数了,但在js中匿名函数用到比较多了,下文我来为各位介绍php中匿名函数用法。
php5.3起就支持匿名函数,可以放心在生产环境中使用,对于php的很多新特性该用的地方要大胆使用,可以使代码更简洁,功能实现也更加方便。
以下是php匿名函数的一个小的简单实例:
function func($list,$u_wangwang)
{
//匿名函数,获取其他行联号
$other_num = function($num){
$num = explode('.', $num);
return $num[0];
};
$res = $other_num($v['sb_other_num']);
return $res;
}
首先在方法里定义了一个匿名函数,有一个参数,然后调用的时候进行传参。
hp中, 传递callback的方式, 一直很丑陋. 在php5.3以前, 我们只有俩种选择:
1. 字符串的函数名
2. 使用create_function的返回值
在php5.3以后, 我们多了一个选择, 也就是closure,
$func = function () { ... };
array_walk($arr, $func);
从实现上来说, 第一种方式: 传递函数名字符串是最简单的.
而第二种方式create_function, 其实和第一种方式本质上一样的, create_function返回一个字符串的函数名, 这个函数名的格式是:
\000_lambda_ . count(anonymous_functions)++;
我们来看看create_function的实现步骤:
1. 获取参数, 函数体
2. 拼凑一个function __lambda_func (参数) { 函数体;} 的字符串
3. eval之
4. 通过__lambda_func在函数表中找到eval后得到的函数体, 找不到就出错
5. 定义一个函数名:\000_lambda_ . count(anonymous_functions)++
6. 用新的函数名替换__lambda_func
7. 返回新的函数名
我们来验证下:
//输出
__lambda_func
因为在eval的时候, 函数名是”__lambda_func”, 所以匿名函数内会输出__lambda_func, 而因为最后用”\000_lambda_” . count(anonymous_functions)++重命名了函数表中的”__lambda_func”函数, 所以可通过”\000_lambda_” . count(anonymous_functions)++调用这个匿名函数.
为了证实这一点, 可以将create_function的返回值dump出来查看.
而在php5.3发布的时候, 其中有一条new feature就是支持闭包/lambda function, 我第一反应是以为zval新增了一个is_function, 但实际上是构造了一个php5.3引入的closure”类”的实例, closure类的构造函数是私有的, 所以不能被直接实例化, 另外closure类是final类, 所以也不能做为基类派生子类.
//php-5.3.0
$class = new reflectionclass(closure);
var_dump($class->isinternal());
var_dump($class->isabstract() );
var_dump($class->isfinal());
var_dump($class->isinterface());
//输出:
bool(true)
bool(false)
bool(true)
bool(false)
?>
而php5.3中对闭包的支持, 也仅仅是把要保持的外部变量, 做为closure对象的”static属性”(并不是普通意义上的可遍历/访问的属性).
//php-5.3.0
$b = laruence;
$func = function($a) use($b) {};
var_dump($func);
/* 输出:
object(closure)#1 (2) {
[static]=>
array(1) {
[b]=>
string(8) laruence
}
[parameter]=>
array(1) {
[$a]=>
string(10)
}
}
*/
这个实现, 个人认为和js对闭包的支持比起来, 还是有些太简陋了~