zoukankan      html  css  js  c++  java
  • [转]php 5.3新增的闭包语法介绍function() use() {}

    From : http://bbs.csdn.net/topics/360002529

    php 5.3新增的闭包语法介绍function() use() {}

    Demo1

    function callback($callback) {
        $callback();
    }
    
    callback(function() {
        print "This is a anonymous function.<br />\n";
    });

    输出: This is a anonymous function.<br />\n
    这里是直接定义一个匿名函数进行传递, 在以往的版本中, 这是不可用的.
    现在, 这种语法非常舒服, 和javascript语法基本一致, 之所以说基本呢, 需要继续向下看
    结论: 一个舒服的语法必然会受欢迎的.

    Demo2

    function callback($callback) {
        $callback();
    }
    $msg = "Hello, everyone";
    $callback = function () use ($msg) {
        print "This is a closure use string value, msg is: $msg. <br />\n";
    };
    $msg = "Hello, everybody";
    callback($callback);

    输出: This is a closure use string value, msg is: Hello, everyone.<br />\n
    这里首先定义了一个闭包, 这次户口本上有名字了...
    use, 一个新鲜的家伙...
    众所周知, 闭包: 内部函数使用了外部函数中定义的变量.
    在PHP新开放的闭包语法中, 我们就是用use来使用闭包外部定义的变量的.
    这里我们使用了外部变量$msg, 定义完之后, 又对其值进行了改变, 闭包被执行后输出的是原始值
    结论: 以传值方式传递的基础类型参数, 闭包use的值在闭包创建是就确定了.

    Demo3

    function callback($callback) {
        $callback();
    }
    
    $msg = "Hello, everyone";
    $callback = function () use (&$msg) {
        print "This is a closure use string value lazy bind, msg is: $msg. <br />\n";
    };
    $msg = "Hello, everybody";
    callback($callback);

    输出: This is a closure use string value lazy bind, msg is: Hello, everybody.<br />\n
    换一种引用方式, 我们使用引用的方式来use
    可以发现这次输出是闭包定义后的值...
    这个其实不难理解, 我们以引用方式use, 那闭包use的是$msg这个变量的地址
    当后面对$msg这个地址上的值进行了改变之后, 闭包内再输出这个地址的值时, 自然改变了.

    Demo4

    function callback($callback) {
        $callback();
    }
    
    $obj = (object) "Hello, everyone";
    $callback = function () use ($obj) {
        print "This is a closure use object, msg is: {$obj->scalar}. <br />\n";
    };
    //var_dump($obj);
    $obj = (object) "Hello, everybody";
    callback($callback);

    输出: This is a closure use object, msg is: Hello, everyone.<br />\n
    闭包中输出的是之前被拷贝的值为Hello, everyone的对象, 后面是对$obj这个名字的一个重新赋值.
    可以这样考虑
    1. obj是对象Hello, everyone的名字
    2. 对象Hello, everyone被闭包use, 闭包产生了一个对Hello, everyone对象的引用
    3. obj被修改为Hello, everybody这个对象的名字
    4. 注意, 是名字obj代表的实体变了, 而不是Hello, everyone对象, 那自然闭包的输出还是前面的Hello, everyone

    Demo5

    function callback($callback) {
        $callback();
    }
    
    $obj = (object) "Hello, everyone";
    $callback = function () use ($obj) {
        print "This is a closure use object, msg is: {$obj->scalar}. <br />\n";
    };
    $obj->scalar = "Hello, everybody";
    callback($callback);

    输出: This is a closure use object, msg is: Hello, everybody.<br />\n
    还是按照上面的步骤, 按部就班的来吧:
    1. obj名字指向Hello, everyone对象
    2. 闭包产生一个引用指向Hello, everyone对象
    3. 修改obj名字指向的对象(即Hello, everyone对象)的scalar值
    4. 执行闭包, 输出的自然是Hello, everybody, 因为其实只有一个真正的对象

    Demo6

    function callback($callback) {
        $callback();
    }
    
    $obj = (object) "Hello, everyone";
    $callback = function () use (&$obj) {
        print "This is a closure use object lazy bind, msg is: {$obj->scalar}. <br />\n";
    };
    $obj = (object) "Hello, everybody";
    callback($callback);

    输出: This is a closure use object lazy bind, msg is: Hello, everybody.<br />\n
    闭包引用的是什么呢? &$obj, 闭包产生的引用指向$obj这个名字所指向的地址.
    因此, 无论obj怎么变化, 都是逃不脱的....
    所以, 输出的就是改变后的值

    Demo7

    function counter() {
        $counter = 1;
        return function() use(&$counter) {return $counter++;};
    }
    $counter1 = counter();
    $counter2 = counter();
    echo "counter1: " . $counter1() . "<br />\n";
    echo "counter1: " . $counter1() . "<br />\n";
    echo "counter1: " . $counter1() . "<br />\n";
    echo "counter1: " . $counter1() . "<br />\n";
    echo "counter2: " . $counter2() . "<br />\n";
    echo "counter2: " . $counter2() . "<br />\n";
    echo "counter2: " . $counter2() . "<br />\n";
    echo "counter2: " . $counter2() . "<br />\n";

    结果:

    counter1: 1
    counter1: 2
    counter1: 3
    counter1: 4
    counter2: 1
    counter2: 2
    counter2: 3
    counter2: 4


    一个利用闭包的计数器产生器
    这里其实借鉴的是python中介绍闭包时的例子...
    我们可以这样考虑:
    1. counter函数每次调用, 创建一个局部变量$counter, 初始化为1.
    2. 然后创建一个闭包, 闭包产生了对局部变量$counter的引用.
    3. 函数counter返回创建的闭包, 并销毁局部变量, 但此时有闭包对$counter的引用,
    它并不会被回收, 因此, 我们可以这样理解, 被函数counter返回的闭包, 携带了一个游离态的
    变量.
    4. 由于每次调用counter都会创建独立的$counter和闭包, 因此返回的闭包相互之间是独立的.
    5. 执行被返回的闭包, 对其携带的游离态变量自增并返回, 得到的就是一个计数器.
    结论: 此函数可以用来生成相互独立的计数器.

  • 相关阅读:
    剑指offer——关于排序算法的应用(一):归并排序
    剑指offer——关于排序算法的应用:选择排序和冒泡排序
    剑指offer:将矩阵选择、螺旋输出矩阵——Python之光
    剑指offer:链表——常见的多指针协同操作:
    剑指Offer:编程习惯篇:代码鲁棒性,代码可扩展性——防御性的编程习惯,解决问题时方法分模块考虑
    剑指offer:数字二进制含1个数,快速幂运算:二进制位运算的运用
    剑指offer:斐波那契数列,跳台阶,变态跳台阶——斐波那契数列类题目:
    回溯法实现各种组合的检索:
    剑指offer:二维数组中查找
    jdk生成https证书的方法
  • 原文地址:https://www.cnblogs.com/Athrun/p/php_closures.html
Copyright © 2011-2022 走看看