关于php匿名函数中的use

来源:互联网 发布:工业进销存软件免费版 编辑:程序博客网 时间:2024/06/06 00:25

http://blog.csdn.net/qmhball/article/details/53668113

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

<?php$msg = [1,2,3];$func = function()use($msg){    print_r($msg);};  $func();?>运行输出Array(    [0] => 1    [1] => 2    [2] => 3)
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

关于继承变量的时机

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

<?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
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

可见,继承变量的行为是在函数定义时产生的。上例中定义funcmsg,所以函数运行时$msg就是未定义变量。


关于use中使用引用传值

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

<?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)
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26

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

<?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        ))
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32

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

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

<?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
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

我们改为使用引用传递

$func = function()use(&$msg){    echo $msg[0],"\n";};运行输出1
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

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


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

<?phpclass 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();?>运行结果01
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27

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

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

public function mkFunc(){    //唯一改动是此处加了use    $func = function()use($this){        echo $this->_num++, "\n";    };    return $func;}运行输出PHP Fatal error:  Cannot use $this as lexical variable 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

修改为

public function mkFunc(){    $self = $this;    $func = function()use($self){        echo $this->_num++, "\n";    };    return $func;}运行结果01
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

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

原创粉丝点击