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;


  • 2018-12-26 16:12:56

    nginx+php-fpm模式php内存泄漏探究

    这里要重点说一下第三步骤。第三步涉及到php-fpm进程生命周期的东西。一个php-fpm的生命周期大致是这样的:模块初始化(MINIT)-> 模块激活(RINIT)-> 请求处理 -> 模块停用(RSHUTDOWN) -> 模块激活(RINIT)-> 请求处理 -> 模块停用(RSHUTDOWN)……. 模块激活(RINIT)-> 请求处理 -> 模块停用(RSHUTDOWN)-> 模块关闭(MSHUTDOWN)。在一个php-fpm进程的生命周期里,会有多次的模块激活(RINIT)-> 请求处理 -> 模块停用(RSHUTDOWN)的过程。这个“请求处理”的大致过程是这样的:php读取相应的php文件,对其进行词法分析,生成opcode,zend虚拟机执行opcode。

  • 2019-01-01 21:38:51

    php使用curl设置超时的重要性

    网站登录不了,原因是没有可用的 PHP 子进程来响应新的请求了。这可能是是由于PHP-curl 没有设置超时时间引起的。

  • 2019-01-01 21:42:34

    php-fpm 启动参数及重要配置详解

    如果file_get_contents请求的远程资源如果反应过慢,file_get_contents就会一直卡在那里不会超时。我们知道php.ini 里面max_execution_time 可以设置 PHP 脚本的最大执行时间,但是,在 php-cgi(php-fpm) 中,该参数不会起效。真正能够控制 PHP 脚本最大执行时间的是 php-fpm.conf 配置文件中的request_terminate_timeout参数。