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;
});

是不是逼格满满呢?

OK,介绍了几个用法~希望对大家有帮助,如果有问题,欢迎指出,如果您喜欢,可以点下推荐~

文章持续更新,欢迎大家关注。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java架构沉思录

关于泛型,你可能不知道的事儿

大家可能会有疑问,我为什么叫做泛型是一个守门者。这其实是我个人的看法而已,我的意思是说泛型没有其看起来那么深不可测,它并不神秘与神奇。泛型是 Java 中一个很...

331
来自专栏進无尽的文章

Swift| 基础语法(二)

总结下 swift下的基础语法,里面涉及到:常量&变量、Swift中的数据类型、逻辑分支、循环、字符串相关、数组和字典、方法的书写调用等内容,考虑到阅读体验分多...

582
来自专栏IMWeb前端团队

递归函数的优化

本文作者:IMWeb 寒纱阁主 原文出处:IMWeb社区 未经同意,禁止转载 递归函数是一个函数自我调用而构成的,如下是一个典型的递归阶乘函数: fu...

17410
来自专栏Java技术栈

揭开Java 泛型类型擦除神秘面纱

大家可能会有疑问,我为什么叫做泛型是一个守门者。这其实是我个人的看法而已,我的意思是说泛型没有其看起来那么深不可测,它并不神秘与神奇。泛型是 Java 中一个很...

654
来自专栏老九学堂

【干货】Java字符串之10大热点问题!

其实作为Java初学者,字符串是一个必须迈过的坎,所以老九君为了大家能够更好的掌握字符串这个知识点,以及深入的了解一些原理,搜罗总结了以下10条Java字符串的...

2554
来自专栏编程

一篇文章告诉你什么是 Python 元类

. 龟叔发明了 Python,然后集成了一堆概念在这门语言里面,比如:迭代器,装饰器,函数,生成器,类,对象,协程等等。 这些概念对初学者似乎没一个好懂的,不过...

17810
来自专栏较真的前端

一些你可能不知道的前端小技巧

1796
来自专栏Java Web

Java学习笔记(2)——数据类型

终于要写点干货了,其实思考了很久下面一篇文章要写什么,主要的纠结点在于,既想要分享那些精美的知识,又怕这些知识不太好嚼。后来想想还是对初学者不太好友算了..一...

2493
来自专栏较真的前端

关于数组的前端面试题,你是否都能答对?

2133
来自专栏CoXie带你学编程

详解Python 2.x 与 Python 3.x 的区别

如果你是刚接触 Python 的初学者,那你可能是直接学习 Python 3.x 版本。对于 Python 2.x 的版本是不会有所接触。官方也宣布在 2020...

782

扫码关注云+社区