【发布时间】:2013-11-11 23:05:53
【问题描述】:
有人能解释一下gc_collect_cycles函数在什么情况下有用吗?是否应该在大量内存使用即将发生之前调用它?
【问题讨论】:
-
似乎 gc_collect_cycles 仅在 GC 被禁用时才真正有用。
标签: php memory memory-management garbage-collection
有人能解释一下gc_collect_cycles函数在什么情况下有用吗?是否应该在大量内存使用即将发生之前调用它?
【问题讨论】:
标签: php memory memory-management garbage-collection
PHP 默认启用“垃圾收集器”。它用于释放“垃圾”使用的内存。
gc_collect_cycles() 强制收集任何现有的垃圾周期。它返回收集(释放)周期(对象、变量值...)的数量。启用的垃圾收集器不时在内部调用此函数以释放资源。在大多数情况下,PHP 脚本的寿命很短。在这种情况下,所有垃圾都将在工作结束时销毁,无需任何垃圾收集。
有时需要手动管理 GC:
gc_disable() 可以加快一些长时间的操作,但也会导致一些内存开销。gc_collect_cycles() 可用于指定 GC 的正确时刻。使用gc_collect_cycles() 的另一个原因 - 调试。假设,您想知道memory_get_usage() 的某些代码块的内存消耗是多少。您需要先禁用 GC,否则您会得到错误的结果。之后,您希望将 GC 和您的应用程序消耗的时间分开。所以打电话给gc_collect_cycles() 并测量前后的时间/内存。
小例子:
class A {
public $ref;
public $name;
public function __construct($name) {
$this->name = $name;
echo($this->name.'->__construct();'.PHP_EOL);
}
public function __destruct() {
echo($this->name.'->__destruct();'.PHP_EOL);
}
}
gc_disable();
$a1 = new A('$a1');
$a2 = new A('$a2');
$a1->ref = $a2;
$a2->ref = $a1;
$b = new A('$b');
$b->ref = $a1;
echo('$a1 = $a2 = $b = NULL;'.PHP_EOL);
$a1 = $a2 = $b = NULL;
echo('gc_collect_cycles();'.PHP_EOL);
echo('// removed cycles: '.gc_collect_cycles().PHP_EOL);
echo('exit();'.PHP_EOL);
将输出:
$a1->__construct();
$a2->__construct();
$b->__construct();
$a1 = $a2 => $b = NULL;
$b->__destruct();
gc_collect_cycles();
$a2->__destruct();
$a1->__destruct();
// removed cycles: 4
这意味着只有$b 在被询问时被销毁。其他$a1 和$a2 有循环引用,它的name 属性也消耗内存。两个对象 + 两个字符串 = 4(被gc_collect_cycles() 删除)。
【讨论】: