PHP中的闭包和匿名函数

PHP中的闭包和匿名函数

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

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

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

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

创建闭包

$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:

$nubmers = array_map(function ($number) {
    return $number++;
}, [1,2,3]);

var_dump($numbers);

附加状态

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

  • 使用use关键字

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

    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类。因此,闭包可以访问绑定闭包的对象中受保护和私有的成员变量。

    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
    
    */
    
posted @ 2017-12-24 18:35  praglody  阅读(462)  评论(0编辑  收藏  举报