PHP 中的Closure

2017-02-24 16:53:58

PHP 中的Closure

Closure,匿名函数,又称为Anonymous functions,是php5.3的时候引入的。匿名函数就是没有定义名字的函数。这点牢牢记住就能理解匿名函数的定义了。

比如下面的代码

function test() {    return 100;
};function testClosure(Closure $callback)
{    return $callback();
}$a = testClosure(test());print_r($a);exit;

这里的test()永远没有办法用来作为testClosure的参数,因为它并不是“匿名”函数。

所以应该改成这样:

$f = function () {    return 100;
};function testClosure(Closure $callback){    return $callback();
}

$a = testClosure($f);
print_r($a);exit;

好,如果要调用一个类里面的匿名函数呢?

class C {    public static function testC() {        return function($i) {            return $i+100;
        };
    }
}

$f = function ($i) {    return $i + 100;
};function testClosure(Closure $callback){    return $callback(13);
}

$a = testClosure(C::testC());
print_r($a);exit;

应该这么写,其中的C::testC()返回的是一个funciton。

绑定的概念

上面的例子的Closure只是全局的的匿名函数,好了,我现在想指定一个类有一个匿名函数。也可以理解说,这个匿名函数的访问范围不再是全局的了,是一个类的访问范围。

那么我们就需要将“一个匿名函数绑定到一个类中”。

<?phpclass A {    public $base = 100;

}class B {    private $base = 1000;
}

$f = function () {    return $this->base + 3;
};


$a = Closure::bind($f, new A);
print_r($a());echo PHP_EOL;

$b = Closure::bind($f, new B , 'B');
print_r($b());echo PHP_EOL;

上面的例子中,ff这个匿名函数中莫名奇妙的有个this,这个this关键词就是说明这个匿名函数是需要绑定在类中的。

绑定之后,就好像A中有这么个函数一样,但是这个函数是public还是private,bind的最后一个参数就说明了这个函数的可调用范围。

对于bindTo,看例子:

<?phpclass A {    public $base = 100;

}class B {    private $base = 1000;
}class C {    private static $base = 10000;
}

$f = function () {    return $this->base + 3;
};

$sf = static function() {    return self::$base + 3;
};


$a = Closure::bind($f, new A);
print_r($a());echo PHP_EOL;

$b = Closure::bind($f, new B , 'B');
print_r($b());echo PHP_EOL;

$c = $sf->bindTo(null, 'C');
print_r($c());echo PHP_EOL;


  • 2019-09-28 08:00:30

    Java.io.tmpdir介绍

    System.getproperty(“java.io.tmpdir”)是获取操作系统缓存的临时目录,不同操作系统的缓存临时目录不一样,

  • 2019-09-28 08:36:43

    Ehcache配置持久化到硬盘,只存储到硬盘

    Ehcache默认配置的话 为了提高效率,所以有一部分缓存是在内存中,然后达到配置的内存对象总量,则才根据策略持久化到硬盘中,这里是有一个问题的,假如系统突然中断运行 那内存中的那些缓存,直接被释放掉了,不能持久化到硬盘;这种数据丢失,对于一般项目是不会有影响的,但是对于我们的爬虫系统,我们是用来判断重复Url的,所以数据不能丢失;

  • 2019-09-28 09:33:18

    put与putIfAbsent区别

    put在放入数据时,如果放入数据的key已经存在与Map中,最后放入的数据会覆盖之前存在的数据, 而putIfAbsent在放入数据时,如果存在重复的key,那么putIfAbsent不会放入值。