在PHP5中,是否保证为每个对象实例调用__destruct()方法?程序中的异常是否可以防止这种情况发生?
值得一提的是,在具有自己的析构函数的子类的情况下,不会自动调用父析构函数.
如果父类执行任何必需的清理,则必须从子类__destruct()方法显式调用parent :: __ destruct().
释放所有引用或脚本终止时将调用析构函数.我认为这意味着脚本正确终止.我会说,关键异常不能保证析构函数被调用.
该PHP文件是有点薄,但它说,在析构函数中的异常会导致一些问题.
根据我的经验,析构函数将始终在PHP 5.3中调用,但要注意,如果某段代码调用exit()或发生致命错误,PHP将以"任意"顺序调用析构函数(我认为实际的顺序是内存或为对象保留内存的顺序.实际上,这个顺序几乎总是有问题的).这在PHP文档中称为"关闭序列".
析构函数的PHP文档说:
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引擎遇到分段错误或发生其他一些内部错误,那么所有的赌注都会被取消.
如果您想了解"关闭序列"的当前实现,请参阅/sf/ask/17360801/.请注意,此实现可能会在将来的PHP版本中更改.
循环引用存在当前错误,这会阻止隐式调用destruct方法.http://bugs.php.net/bug.php?id=33595 应该在5.3中修复
如果你想确定,请使用关机功能:register_shutdown_function()