PHP回调函数与匿名函数实例详解


Posted in PHP onAugust 16, 2017

本文实例讲述了PHP回调函数与匿名函数。分享给大家供大家参考,具体如下:

回调函数和匿名函数

回调函数、闭包在JS中并不陌生,JS使用它可以完成事件机制,进行许多复杂的操作。PHP中却不常使用,今天来说一说PHP中中的回调函数和匿名函数。

回调函数

回调函数:Callback (即call then back 被主函数调用运算后会返回主函数),是指通过函数参数传递到其它代码的,某一块可执行代码的引用。

通俗的解释就是把函数作为参数传入进另一个函数中使用;PHP中有许多 “需求参数为函数” 的函数,像array_map,usort,call_user_func_array之类,他们执行传入的函数,然后直接将结果返回主函数。好处是函数作为值使用起来方便,而且代码简洁,可读性强。

匿名函数

匿名函数,顾名思义,是没有一个确定函数名的函数,PHP将匿名函数和闭包视作相同的概念(匿名函数在PHP中也叫作闭包函数)。它的用法,当然只能被当作变量来使用了。

PHP中将一个函数赋值给一个变量的方式有四种:

① 我们经常会用到的:函数在外部定义/或PHP内置,直接将函数名作为字符串参数传入。注意:如果是类静态函数的话以CLASS::FUNC_NAME的方式传入。

② 使用create_function($args, $func_code);创建函数,会返回一个函数名。 $func_code为代码体,$args为参数字符串,以','分隔;

③ 直接赋值:$func_name = function($arg){statement}

④ 直接使用匿名函数,在参数处直接定义函数,不赋给具体的变量值;

第一种方式因为是平常所用,不再多提;第二种类似eval()方法的用法,也被PHP官方列为不推荐使用的方式,而且其定义方式太不直观,我除了测试外,也没有在其他地方使用过,也略过不提。在这里重点说一下第三种和第四种用法;

后两种创建的函数就被称为匿名函数,也就是闭包函数, 第三种赋值法方式创建的函数非常灵活,可以通过变量引用。可以用 is_callable($func_name) 来测试此函数是否可以被调用, 也可以通过$func_name($var)来直接调用;而第四种方式创建的函数比较类似于JS中的回调函数,不需要变量赋值,直接使用;

另外要特别介绍的是 use 关键词,它可以在定义函数时,用来引用父作用域中的变量;用法为 function($arg) use($outside_arg) {function_statement} 。其中$outside_arg 为父作用域中的变量,可以在function_statement使用。

这种用法用在回调函数“参数值数量确定”的函数中。 如usort需求$callback的参数值为两项,可是我们需要引入别的参数来影响排序怎么办呢?使用use()关键词就很方便地把一个新的变量引入$callback内部使用了。

array_map/array_filter/array_walk:

把这三个函数放在一块是因为这三个函数在执行逻辑上比较类似,类似于下面的代码:

$result = [];
foreach($vars as $key=>$val){
  $item = callback();
  $result[] = $item;
}
return $result;
array_walk($vars, $callback)

其callback应如下:

$callback = function(&$val, $key[, $arg]){  
  doSomething($val);
}

array_walk返回执行是否成功,是一个布尔值。对$value添加引用符号可以在函数内改变$value值,以达到改变$vars数组的效果。由于其$callback对参数数量要求为两项,array_walk不能传入strtolower/array_filter之类的$callback,若想实现类似功能,可以使用接下来要说的array_map()

array_walk_recursive($arr, $callback);

返回值和执行机制类似于array_walk;

其callback同array_walk,不同的是,如果$val是数组,函数会递归地向下处理$val;需要注意的是这样的话$val为数组的$key就会被忽略掉了。

array_filter($vars, $callback, $flag);

其$callback类似于:

$callback = function($var){
  return true or false;     
}

array_filter会过滤掉$callback执行时返回为false的项目,array_filter返回过滤完成后的数组。

第三个参数 $flag决定其callback形参$var的值,不过这个可能是PHP高版本的特性,我的PHP5.5.3不支持,大家可以自行测试。默认传入数组每项的value,当flag为ARRAY_FILTER_USE_KEY传入数组每项的key,ARRAY_FILTER_USE_BOTH传入键和值;

array_map($callback, &$var_as [,$var_bs...]);

其$callback类似于:

$callback = function($var_a[, $var_b...]){
  doSomething($var_a, $var_b);
}

返回$var_as经过callback处理后的数组(会改变原数组);如果有多个数组的时候将两个数组同样顺序的项目传入处理,执行次数为参数数组中项目最多的个数;

usort/array_reduce

把这两个函数放在一块,因为他们的执行机制都有些特殊。

usort(&$vars, $callback)

$callback应该如下:

callback = function($left, $right){
    $res = compare($left, $right);
    return $res;
}

usort返回执行成功与否,bool值。用户自定义方法 比较$left 和 $right,其中$left和$right是$vars中的任意两项;

$left > $right时返回 正整数, $left < $right时返回 负整数, $left = $right时返回0;

$vars中的元素会被取出会被由小到大升序排序。 想实现降序排列,将$callback的返回值反一下就行了。

array_reduce($vars ,$callable [, mixed $initial = NULL])

$callback应该如下:

$callback = function($initial, $var){
    $initial = calculate($initail, $var);
    return $initial;
}

初始值$initial默认为null,返回经过迭代后的initial;一定要将$initial返回,这样才能不停地改变$initial的值,实现迭代的效果。

这里顺便说一下map和reduce的不同:

map:将数组中的成员遍历处理,每次返回处理后的一个值,最后结果值为所有处理后值组成的多项数组;

reduce:遍历数组成员,每次使用数组成员结合初始值处理,并将初始值返回,即使用上一次执行的结果,配合下一次的输入继续产生结果,结果值为一项;

call_user_func/call_user_func_array

call_user_func[_array]($callback, $param)

$callback形如:

$callback = function($param){
    $result = statement(); 
    return $result;
}

返回值多种,具体看$callback。

可用此函数实现PHP的事件机制,其实并不高深,在判断条件达成,或程序执行到某一步后 call_user_func()就OK了。这个我在之前的博客中也有介绍到:搭建自己的PHP框架

总结

其实以上$callback不用单独定义并使用变量引用,使用上面说过的第四种函数定义方式,直接在函数内定义,使用‘完全'匿名函数就行了。 如:

usort($records, function mySortFunc($arg) use ($order){
  func_statement;
});

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
PHP开发需要注意的安全问题
Sep 01 PHP
PHP APC缓存配置、使用详解
Mar 06 PHP
ThinkPHP3.1新特性之动态设置自动完成及自动验证示例代码
Jun 23 PHP
PHP中的switch语句的用法实例详解
Oct 21 PHP
Zend Framework教程之Zend_Db_Table用法详解
Mar 21 PHP
深入理解PHP 数组之count 函数
Jun 13 PHP
Yii2.0预定义的别名功能小结
Jul 04 PHP
PHP导出带样式的Excel示例代码
Aug 28 PHP
PHP基于mssql扩展远程连接MSSQL的简单实现方法
Oct 08 PHP
PHP常用算法和数据结构示例(必看篇)
Mar 15 PHP
php实现的AES加密类定义与用法示例
Jan 29 PHP
php实现小程序支付完整版
Oct 09 PHP
搭建自己的PHP MVC框架详解
Aug 16 #PHP
Laravel使用支付宝进行支付的示例代码
Aug 16 #PHP
laravel 中如何使用ajax和vue总结
Aug 16 #PHP
yii gridview实现时间段筛选功能
Aug 15 #PHP
PHP使用GD库制作验证码的方法(点击验证码或看不清会刷新验证码)
Aug 15 #PHP
Laravel学习教程之IOC容器的介绍与用例
Aug 15 #PHP
Laravel 5.4向IoC容器中添加自定义类的方法示例
Aug 15 #PHP
You might like
Chrome Web App开发小结
2014/09/04 PHP
如何解决PHP无法实现多线程的问题
2015/09/25 PHP
Yii2 hasOne(), hasMany() 实现三表关联的方法(两种)
2017/02/15 PHP
PHP+Ajax实现的博客文章添加类别功能示例
2018/03/29 PHP
php和js实现根据子网掩码和ip计算子网功能示例
2019/11/09 PHP
几个比较经典常用的jQuery小技巧
2010/03/01 Javascript
js 调用百度地图api并在地图上进行打点添加标注
2014/05/13 Javascript
jQuery实现table隔行换色和鼠标经过变色的两种方法
2014/06/15 Javascript
jQuery实现冻结表头的方法
2015/03/09 Javascript
在AngularJS应用中实现一些动画效果的代码
2015/06/18 Javascript
jquery实现弹窗功能(窗口居中显示)
2017/02/27 Javascript
vue2.0实战之基础入门(1)
2017/03/27 Javascript
Vue.extend构造器的详解
2017/07/17 Javascript
使用mpvue搭建一个初始小程序及项目配置方法
2018/12/03 Javascript
微信小程序学习笔记之表单提交与PHP后台数据交互处理图文详解
2019/03/28 Javascript
详解JavaScript的内存空间、赋值和深浅拷贝
2019/04/17 Javascript
Vue 指令实现按钮级别权限管理功能
2019/04/23 Javascript
使用ThinkJs搭建微信中控服务的实现方法
2019/08/08 Javascript
JS严格模式原理与用法实例分析
2020/04/27 Javascript
Vue项目打包编译优化方案
2020/09/16 Javascript
vue中使用echarts的示例
2021/01/03 Vue.js
[01:47]2018年度DOTA2最佳教练-完美盛典
2018/12/16 DOTA
使用Python编写Linux系统守护进程实例
2015/02/03 Python
Python环境下安装使用异步任务队列包Celery的基础教程
2016/05/07 Python
利用Python如何实现数据驱动的接口自动化测试
2018/05/11 Python
Python unittest单元测试框架总结
2018/09/08 Python
python文件选择对话框的操作方法
2019/06/27 Python
opencv-python的RGB与BGR互转方式
2020/06/02 Python
美国的Eastbay旗下的运动款子品牌:Final-Score
2018/01/01 全球购物
美国摩托车头盔、零件、齿轮及配件商店:Cycle Gear
2019/06/12 全球购物
英国户外服装、鞋类和设备的领先零售商:Millets
2020/10/12 全球购物
如果让你测试一台高速激光打印机,你都会进行哪些测试
2012/12/04 面试题
JAVA高级程序员面试题
2013/09/06 面试题
关于运动会的口号
2014/06/07 职场文书
学校开除通知书
2015/04/25 职场文书
大学运动会通讯稿
2015/07/18 职场文书