关于php匿名函数中的use 闭包(Closure)

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

https://www.cnblogs.com/xiaorenwu702/p/6212697.html

匿名函数中的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 /search/ballqiu/c.php on line 4

可见,继承变量的行为是在函数定义时产生的。上例中定义func时,没有找到外部的func时,没有找到外部的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],"\n";
};

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

运行输出
PHP Notice:  Undefined variable: msg

我们改为使用引用传递

$func = function()use(&$msg){
    echo $msg[0],"\n";
};

运行输出
1

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


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

<?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改为

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

    return $func;
}

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

修改为

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

    return $func;
}

运行结果
0
1

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

不知不觉发现PHP已经出到了5.5版本,而自己一直在用PHP5.2,让我看起来像深山出来的小伙子一样,又土又落后。在我习惯在javascript中使用闭包之后,忽然间对PHP的闭包打起了兴趣。

于是乎在网上下了个WAMP集成开发环境,是PHP5.3版本的(PHP5.3开始引入了闭包的特性),不得不说WAMP安装使用真的很方便。简单配置了一下,开始动手。

匿名函数

提到闭包就不得不想起匿名函数,也叫闭包函数(closures),貌似PHP闭包实现主要就是靠它。声明一个匿名函数是这样:

$func = function() {
    
}; //带结束符

可以看到,匿名函数因为没有名字,如果要使用它,需要将其返回给一个变量。匿名函数也像普通函数一样可以声明参数,调用方法也相同:

$func = function( $param ) {
    echo $param;
};

$func( 'some string' );

//输出:
//some string

顺便提一下,PHP在引入闭包之前,也有一个可以创建匿名函数的函数:create function,但是代码逻辑只能写成字符串,这样看起来很晦涩并且不好维护,所以很少有人用。

实现闭包

将匿名函数在普通函数中当做参数传入,也可以被返回。这就实现了一个简单的闭包。

下边有三个例子

//例一
//在函数里定义一个匿名函数,并且调用它
function printStr() {
    $func = function( $str ) {
        echo $str;
    };
    $func( 'some string' );
}

printStr();



//例二
//在函数中把匿名函数返回,并且调用它
function getPrintStrFunc() {
    $func = function( $str ) {
        echo $str;
    };
    return $func;
}

$printStrFunc = getPrintStrFunc();
$printStrFunc( 'some string' );




//例三
//把匿名函数当做参数传递,并且调用它
function callFunc( $func ) {
    $func( 'some string' );
}

$printStrFunc = function( $str ) {
    echo $str;
};
callFunc( $printStrFunc );

//也可以直接将匿名函数进行传递。如果你了解js,这种写法可能会很熟悉
callFunc( function( $str ) {
    echo $str;
} );

连接闭包和外界变量的关键字:USE

闭包可以保存所在代码块上下文的一些变量和值。PHP在默认情况下,匿名函数不能调用所在代码块的上下文变量,而需要通过使用use关键字。

换一个例子看看:

function getMoney() {
    $rmb = 1;
    $dollar = 6;
    $func = function() use ( $rmb ) {
        echo $rmb;
        echo $dollar;
    };
    $func();
}

getMoney();

//输出:
//1
//报错,找不到dorllar变量

可以看到,dollar没有在use关键字中声明,在这个匿名函数里也就不能获取到它,所以开发中要注意这个问题。

有人可能会想到,是否可以在匿名函数中改变上下文的变量,但我发现是不可以的:

function getMoney() {
    $rmb = 1;
    $func = function() use ( $rmb ) {
        echo $rmb;
        //把$rmb的值加1
        $rmb++;
    };
    $func();
    echo $rmb;
}

getMoney();

//输出:
//1
//1

啊,原来use所引用的也只不过是变量的一个副本而已。但是我想要完全引用变量,而不是复制。

要达到这种效果,其实在变量前加一个 & 符号就可以了:

function getMoney() { $rmb = 1; $func = function() use ( &$rmb ) { echo $rmb; //把$rmb的值加1 $rmb++; }; $func(); echo $rmb; } getMoney(); //输出: //1 //2

好,这样匿名函数就可以引用上下文的变量了。如果将匿名函数返回给外界,匿名函数会保存use所引用的变量,而外界则不能得到这些变量,这样形成‘闭包’这个概念可能会更清晰一些。

根据描述改变一下上面的例子:

function getMoneyFunc() {
    $rmb = 1;
    $func = function() use ( &$rmb ) {
        echo $rmb;
        //把$rmb的值加1
        $rmb++;
    };
    return $func;
}

$getMoney = getMoneyFunc();
$getMoney();
$getMoney();
$getMoney();

//输出:
//1
//2
//3

猜你喜欢

转载自blog.csdn.net/weixin_37657720/article/details/81087011
今日推荐