前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >关于php匿名函数中的use

关于php匿名函数中的use

作者头像
跑马溜溜的球
发布2020-12-07 15:12:25
9570
发布2020-12-07 15:12:25
举报
文章被收录于专栏:日积月累1024

匿名函数中的use,其作用就是从父作用域继承变量。 下例是最常见的用法,如果不使用use,函数中将找不到变量$msg。

代码语言:javascript
复制
<?php
$msg = [1,2,3];
$func = function()use($msg){
    print_r($msg);
};  

$func();
?>

运行输出
Array
(
    [0] => 1
    [1] => 2
    [2] => 3
)

关于继承变量的时机

继承变量的行为是在函数定义时产生还是在函数调用时产生?我们调整下上例中代码的顺序,将$msg置于函数定义之后。

代码语言:javascript
复制
<?php
$func = function()use($msg){
    print_r($msg);
};  

$msg = [1,2,3];

$func();
?>

运行输出
PHP Notice:  Undefined variable: msg in /search/ballqiu/c.php on line 4

可见,继承变量的行为是在函数定义时产生的。上例中定义 func时,没有找到外部的 func时,没有找到外部的msg,所以函数运行时$msg就是未定义变量。


关于use中使用引用传值

我们知道,在匿名函数的use中如果使用引用传值,那么匿名函数中对参数值的改变会同样影响外部相应变量。比如下面的例子:

代码语言:javascript
复制
<?php
$msg = [1,2,3];
$func = function()use(&$msg){
    $msg[0]++;
    print_r($msg);
};


$func();

print_r($msg);
?>

运行输出
Array
(
    [0] => 2
    [1] => 2
    [2] => 3
)
Array
(
    [0] => 2
    [1] => 2
    [2] => 3
)

那么是不是任何情况下,想通过匿名函数改变外部变量值都一定要通过引用方式向use传值呢?看下面这个例子:

代码语言:javascript
复制
<?php
$msg = new ArrayObject([1,2,3], ArrayObject::ARRAY_AS_PROPS);
$func = function()use($msg){
    $msg[0]++;
    print_r($msg);
};

$func();
print_r($msg);
?>

运行输出
ArrayObject Object
(
    [storage:ArrayObject:private] => Array
        (
            [0] => 2
            [1] => 2
            [2] => 3
        )

)
ArrayObject Object
(
    [storage:ArrayObject:private] => Array
        (
            [0] => 2
            [1] => 2
            [2] => 3
        )

)

可见,如果传递object类型的变量,即使不显示使用引用传递,匿名函数中变量值的改变同样会影响到外部相关变量。

但是,问题又来了。向use传递object变量时,使用引用与不使用引用到底有没有区别呢?还是来看例子

代码语言:javascript
复制
<?php
$func = function()use($msg){
    echo $msg[0],"\n";
};

$msg = new ArrayObject([1,2,3], ArrayObject::ARRAY_AS_PROPS);
$func();
?>

运行输出
PHP Notice:  Undefined variable: msg

我们改为使用引用传递

代码语言:javascript
复制
$func = function()use(&$msg){
    echo $msg[0],"\n";
};

运行输出
1

可见使用引用传递时,即使变量滞后于函数定义,函数内部还是可以找到外部相应的变量,不会出现变量未定义的情况。两者还是有区别的。


关于class中匿名函数里的this及use

代码语言:javascript
复制
<?php
class C{
    protected $_num = 0;

    public function mkFunc(){
        $func = function(){
            echo $this->_num++, "\n";
        };

        return $func;
    }

    public function get(){
        echo $this->_num,"\n";
    }
}

$obj = new C();
$func = $obj->mkFunc();
$func();

$obj->get();
?>

运行结果
0
1

可见匿名函数里的this就是指当前对象,不需要使用use就可以直接找到。

还是上面的例子,如果一定要使用use会是什么效果呢? 将mkFunc改为

代码语言:javascript
复制
public function mkFunc(){
    //唯一改动是此处加了use
    $func = function()use($this){
        echo $this->_num++, "\n";
    };

    return $func;
}

运行输出
PHP Fatal error:  Cannot use $this as lexical variable 

修改为

代码语言:javascript
复制
public function mkFunc(){
    $self = $this;
    $func = function()use($self){
        echo $this->_num++, "\n";
    };

    return $func;
}

运行结果
0
1

可见是否使用use,效果是一样的。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2016/12/15 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 关于继承变量的时机
  • 关于use中使用引用传值
  • 关于class中匿名函数里的this及use
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档