• 浅析PHP中的闭包和匿名函数


    闭包是指在创建时封装周围状态的函数。即使闭包所在的环境不存在了,闭包中封装的状态依然存在。

    匿名函数就是没有名称的函数。匿名函数可以赋值给变量,还能像其他任何PHP对象那样传递。不过匿名函数仍是函数,因此可以调用,还可以传入参数。匿名函数特别适合作为函数或方法的回调。

    注意:理论上讲,闭包和匿名函数是不同的概念。不过,PHP将其视作相同的概念。所以,我们提到闭包时,指的也是匿名函数,反之亦然。

    PHP闭包和匿名函数使用的句法与普通函数相同,但闭包和匿名函数其实是 伪装成函数的对象(Closure类的实例) 。

    创建闭包

    1

    2

    3

    4

    5

    6

    $closure = function($name){

      return sprintf("Hello %s", $name);

    }

    echo $closure("jerry");

    // 检测$closure变量是否是一个闭包

    var_dump($closure instanceof Closure);

    以上代码创建了一个闭包对象,然后将其赋值给 $closure 变量。闭包和普通的PHP函数很像,使用的句法相同,也接收参数,而且能返回值。

    说明:我们之所以能够调用 $closure 变量,是因为这个变量的值是一个闭包,而且闭包对象实现了 __invoke() 魔术方法。只要变量名后有 () ,PHP就会查找并调用 __invoke() 方法。

    使用闭包

    我们通常把PHP闭包当做当做函数和方法的回调使用。很多PHP函数都会用到回调函数,例如 array_map() 和 preg_replace_callback() .如下示例,我们将用 array_map() 处理数组,将数组每一项自增1:

    1

    2

    3

    4

    $nubmers = array_map(function($number){

      return $number++;

    }, [1,2,3]);

    var_dump($numbers);

    附加状态

    PHP闭包不会像真正的javascrypt闭包那样自动封装应用的状态,我们必须手动调用闭包对象的 bindTo() 方法或者使用 use 关键字,把状态附加到PHP闭包上。

    使用 use 关键字

    使用 use 关键字来附加闭包状态更加常见,因此我们先来看这种方式。使用 use 关键字把变量附加闭包上时,附加的变量会记住附加时赋给它的值。

    1

    2

    3

    4

    5

    6

    7

    8

    9

    10

    function Car ($name){

      return function($statu)use($name){

        return sprintf("Car %s is %s", $name, $statu);

      }

    }

    // 将车名封装在闭包中

    $car = Car("bmw");

    // 调用车的动作

    // 输出--> "bmw is running"

    echo $car("running");

    注意:使用 use 关键字可以把多个参数传入闭包,此时要像PHP函数或方法的参数一样,使用逗号分隔多个参数。

    使用 bindTo() 方法附加闭包的状态

    与其它PHP对象类似,每个闭包实例都可以使用 $this 关键字获取闭包的内部状态。闭包对象的默认状态没什么用,不过有一个 __invoke() 魔术方法和 bindTo() 方法。

    bindTo() 方法为闭包增加了一些有趣的潜力。我们可以使用这个方法把 Closure 对象的内部状态绑定到其它对象上。

    bindTo() 方法的第二个参数很重要,其作用是指定绑定闭包的那个对象所属的PHP类。因此,闭包可以访问绑定闭包的对象中受保护和私有的成员变量。

    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

    class TestClosure

    {

      private $name=[];

      private $age;

      private $sex;

      public function addPerson($name, $personCallback){

        // 将闭包对象绑定当前实例

        $this->name[$name] = $personCallback->bindTo($this, __CLASS__);

      }

      public function display($name){

        foreach ($this->name as $key => $callback){

          if($key == $name){

            // 执行闭包对象,将闭包状态附加到类

            $callback();

          }

        }

        echo "name : {$name}\n";

        echo "age : {$this->age}\n";

        echo "sex : {$this->sex}\n";

      }

    }

    $person = new TestClosure();

    $person->addPerson("jerry", function(){

      $this->age = 19;

      $this->sex = "man";

    });

    $person->display("jerry");

    /** output

    name : jerry

    age : 19

    sex : man

    */

  • 相关阅读:
    【股票价格预测】基于改进莱维飞行和混沌映射的粒子群优化BP神经网络预测股票价格研究(Matlab代码实现)
    Java+SpringBoot+Vue.js全栈实践:手机销售网站开发记
    【英语:基础进阶_听口实战运用】D5.听力对话训练
    Shiro授权&&Shiro的注解式开发
    vue钩子函数以及例子
    如何安装React的第一个脚手架
    工地木模板多少钱一张?
    git分布式管理-头歌实验合并远程分支、rebase、储藏
    浅谈C++|模板篇
    shell 脚本语句
  • 原文地址:https://blog.csdn.net/sinat_40572875/article/details/128030127