php的匿名函数和闭包函数

时间:2021-02-04 19:12:13

php的匿名函数和闭包函数

tags: 匿名函数 闭包函数 php闭包函数 php匿名函数 function use


引言:匿名函数和闭包函数都不是特别高深的知识,但是很多刚入门的朋友却总是很困惑,因为大家习惯上写了函数就是用来调用的,匿名函数和闭包函数做什么用的?

匿名函数

php官方解释如下:
匿名函数(Anonymous functions),也叫闭包函数(closures),允许 临时创建一个没有指定名称的函数。最经常用作回调函数(callback)参数的值。当然,也有其它应用的情况。

这里说的很简单,可以用做回调函数,下面例子就是作为回调函数

<?php
echo preg_replace_callback('~-([a-z])~', function ($match) {
    return strtoupper($match[1]);
}, 'hello-world');
// 输出 helloWorld
?>

上述例子中,preg_replace_callback 函数需要三个参数,第一个参数是一个正则表达式用来匹配数据,第二个参数是一个函数,第三个参数是需要匹配的字符串,也可以像下面这样写

<?php
$testFunc = function ($match) {
        return strtoupper($match[1]);
    };

echo preg_replace_callback('~-([a-z])~', $testFunc, 'hello-world');
// 输出 helloWorld
?>

但是我们看到这个方法显然我们只需要用一次,所以没有必要再给他去命名,也没有多大的必要再去把他赋值给一个变量(赋值给变量的过程:PHP 会自动把此种表达式转换成内置类 Closure 的对象实例赋值给变量)

闭包函数

php中匿名函数也叫闭包函数,所以没什么区别。但是一般意义上的闭包的概念不是这样的,我们先说一下传统意义上的闭包,并提供一篇文章中js中的闭包用来和php中的闭包(匿名函数)进行对比。
学习Javascript闭包(Closure)---阮一峰

php中其实没有传统意义上的闭包函数,因为php中的函数不能调用夫作用域中的变量。如下

<?php
    function a(){
        $a = 11;
        function b(){
            $b = 22;
            echo $a;
            echo $b;
        }
        b();
    }
    a();
//报Notice:Undefined variable: a in index.php on line 6
//22
?>

而看上面文章中得知,js中是可以的。所以php中匿名函数也叫闭包函数,也可以传递父作用域的变量进闭包函数(达到了类似js获取父作用域变量的效果),php中使用的是use关键字,具体如下

<?php
$count = 0;

$a = function()
{ var_dump($count); };

$b = function() use ($count)
{ var_dump($count); };

$count++;

$c = function() use (&$count)
{ var_dump($count); };

$count++;

$a();    // null Notice: Undefined variable: count in
$b();    // int 0
$c();    // int 2
$count++;
$b();    // int 0

?>

上述例子的输出和你认为的是不是不同呢。

解析:闭包函数(匿名函数)使用use获取函数定义时的副作用域变量,不管在何时调用。如果要获取调用时的变量值,需要用引用传递。具体怎么使用就要看什么使用场景了。

列举了几种常用的场景

  1. 作为回调函数
//一个我们使用过的例子
<?php
    /*
 * 菜谱拆分食物后的拼接
 * 参数均不能为空
 * */
    public function mergeFoodsStr($str,array $mapping){
// $str = '白菜半棵、__2__鲍菇两只、__0__一根,__1__两根,三者比例为100:100:15,酱油5克,香油2克,盐1克。';
// $mapping = array(
// 0 =>array('name' => '胡萝卜','id' => '81' ),
// 1 =>array ( 'name' => '萝卜', 'id' => '72'),
// 2 =>array ( 'name' => '杏', 'id' => '1841')
// );
        if(empty($str) || empty($mapping)){
            return false;
        }
        $strNew = preg_replace_callback('"|__(\d)__|" ',function ($matches) use ($mapping){
            return $mapping[$matches[1]]['name'];
        },$str);
        $this->log('拼接后的食材字符串',$strNew);
        return $strNew;
    }
?>
   

2.如果是一次性使用的函数,不能被重用,可以使用闭包函数

  1. 使用array_work()函数配合匿名函数来减少foreach层数
  2. 等等