zoukankan      html  css  js  c++  java
  • php 匿名函数中的use

    匿名函数(Anonymous functions),也叫闭包函数(closures),

    允许 临时创建一个没有指定名称的函数。最经常用作回调函数 callable参数的值。当然,也有其它应用的情况。

    匿名函数目前是通过 Closure 类来实现的。

    闭包可以从父作用域中继承变量,任何此类变量都应该用 use 语言结构传递进去。

    PHP 7.1 起,不能传入此类变量: superglobals、 $this 或者和参数重名。

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

    <?php
    $msg = [1,2,3];
    $func = function()use($msg){
        print_r($msg);
    };  
    
    $func();
    ?>
    
    运行输出
    Array
    (
        [0] => 1
        [1] => 2
        [2] => 3
    )

    关于继承变量的时机

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

    <?php
    $func = function()use($msg){
        print_r($msg);
    };  
    
    $msg = [1,2,3];
    
    $func();
    ?>
    
    运行输出
    PHP Notice:  Undefined variable: msg in test.php on line 4

    可见,继承变量的行为是在函数定义时产生的。上例中定义funcfunc时,没有找到外部的msg,所以函数运行时$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
    )

    那么是不是任何情况下,想通过匿名函数改变外部变量值都一定要通过引用方式向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
            )
    
    )

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

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

    <?php
    $func = function()use($msg){
        echo $msg[0],"
    ";
    };
    
    $msg = new ArrayObject([1,2,3], ArrayObject::ARRAY_AS_PROPS);
    $func();
    ?>
    
    运行输出
    PHP Notice:  Undefined variable: msg

    我们改为使用引用传递

    $func = function()use(&$msg){
        echo $msg[0],"
    ";
    };
    
    运行输出
    1

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

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

    <?php
    class C{
        protected $_num = 0;
    
        public function mkFunc(){
            $func = function(){
                echo $this->_num++, "
    ";
            };
    
            return $func;
        }
    
        public function get(){
            echo $this->_num,"
    ";
        }
    }
    
    $obj = new C();
    $func = $obj->mkFunc();
    $func();
    
    $obj->get();
    ?>
    
    运行结果
    0
    1

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

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

    public function mkFunc(){
        //唯一改动是此处加了use
        $func = function()use($this){
            echo $this->_num++, "
    ";
        };
    
        return $func;
    }
    
    运行输出
    PHP Fatal error:  Cannot use $this as lexical variable 

    修改为

    public function mkFunc(){
        $self = $this;
        $func = function()use($self){
            echo $this->_num++, "
    ";
        };
    
        return $func;
    }
    
    运行结果
    0
    1

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

  • 相关阅读:
    Java 定时任务
    Swing中常用的方法
    瀑布流ajax分页
    权限管理
    图片自动加水印
    jsp自动生成静态页(提供思路)
    分页查询语句
    使用phpmailer发送邮件(以QQ邮箱为例)
    在浏览器的标签页显示网站标志图标(或指定图标)的方法
    随机生成验证码
  • 原文地址:https://www.cnblogs.com/ryanzheng/p/15113188.html
Copyright © 2011-2022 走看看