我可以信任要调用PHP __destruct()方法吗?

时间:2008-09-30 03:29:55

标签: php garbage-collection

在PHP5中,是否保证为每个对象实例调用__destruct()方法?程序中的异常是否可以防止这种情况发生?

5 个答案:

答案 0 :(得分:41)

还值得一提的是,对于具有自己的析构函数的子类,父析构函数是自动调用。

如果父类执行任何必需的清理,则必须从子类 __ destruct()方法显式调用 parent :: __ destruct()

答案 1 :(得分:36)

当释放所有引用或脚本终止时,将调用析构函数。我认为这意味着脚本正确终止。我会说,关键异常不能保证析构函数被调用。

PHP documentation有点薄,但确实说析构函数中的异常会导致问题。

答案 2 :(得分:12)

根据我的经验,析构函数将始终在PHP 5.3中调用,但要注意,如果某段代码调用exit()或发生致命错误,PHP将以“任意”顺序调用析构函数(我认为实际的顺序是内存中的顺序还是为对象保留内存的顺序。实际上,这个顺序几乎总是有问题的。这在PHP文档中称为“关闭序列”。

PHP documentation of destructors说:

  

PHP 5引入了类似于其他面向对象语言(如C ++)的析构函数概念。只要没有对特定对象的其他引用,或者在关闭序列期间以任何顺序,就会调用析构函数方法。

因此,如果你有一个类X来保存对Y的引用,那么可以在已经调用Y的析构函数之后调用X的析构函数。希望,对Y的引用并不那么重要......官方说这不是一个错误,因为它已被记录。

然而,解决这个问题非常困难,因为官方PHP无法知道是否正常调用析构函数(析构函数是否按正确顺序调用),或者以“任何”顺序调用析构函数,而不能使用引用对象中的数据因为那些可能已经被摧毁了。可以使用debug_backtrace()并检查堆栈来解决这种缺乏检测的问题。缺乏正常的堆栈似乎暗示了PHP 5.3的“关闭序列”,但这也是未定义的。如果你有循环引用,那么这些对象的析构函数根本不会被PHP 5.2或更小版本调用,并且在PHP 5.3或更高版本的“shutdown sequence”期间将以“any”顺序调用。对于循环引用,不存在逻辑上“正确”的顺序,因此“任何”顺序对那些顺序是有利的。

有一些例外(毕竟这是PHP):

  • 如果在另一个析构函数中调用exit(),则不会调用任何剩余的析构函数(http://php.net/manual/en/language.oop5.decon.php
  • 如果FATAL错误发生在任何地方(许多可能的原因,例如尝试从任何其他析构函数抛出异常可能是一个原因),将不会调用任何剩余的析构函数。

当然,如果PHP引擎遇到分段错误或发生其他一些内部错误,那么所有的赌注都会关闭。

如果您想了解“关闭序列”的当前实现,请参阅https://stackoverflow.com/a/14101767。请注意,此实现可能会在将来的PHP版本中更改。

答案 3 :(得分:10)

循环引用存在当前错误,这会阻止隐式调用destruct方法。 http://bugs.php.net/bug.php?id=33595 它应该在5.3中修复

答案 4 :(得分:8)

如果您想确定,请使用关机功能:register_shutdown_function()