是什么使得__destruct在如此简单的PHP代码中被调用两次?


What makes __destruct called twice in such a simple PHP code?

<?php
class A
{
    static private $_instance = null;
    static public function Init()
    {   
        self::$_instance = new A();
    }   
    function __construct()
    {   
        echo "__construct'n";
    }   
    function __destruct()
    {   
        var_dump(debug_backtrace());
        echo "__destruct'n";
    }   
}
$a = A::Init();

通常,我们应该得到以下输出:(Yes。我在两个不同的服务器(PHP 5.2.10-2ubuntu6.10和PHP 5.3.1)上得到了这个结果

__construct
array(1) {
  [0]=>
  array(5) {
    ["function"]=>
    string(10) "__destruct"
    ["class"]=>
    string(1) "A"
    ["object"]=>
    object(A)#1 (0) {
    }
    ["type"]=>
    string(2) "->"
    ["args"]=>
    array(0) {
    }
  }
}
__destruct

但是,在另一台使用CentOS版本5.7和PHP 5.2.17的服务器上,我得到了这个:

    __construct
    array(2) {
      [0]=>
      array(7) {
        ["file"]=>
        string(10) "/tmp/1.php"
        ["line"]=>
        int(7)
        ["function"]=>
        string(10) "__destruct"
        ["class"]=>
        string(1) "A"
        ["object"]=>
        object(A)#1 (0) {
        }
        ["type"]=>
        string(2) "->"
        ["args"]=>
        array(0) {
        }
      }
      [1]=>
      array(6) {
        ["file"]=>
        string(10) "/tmp/1.php"
        ["line"]=>
        int(21)
        ["function"]=>
        string(4) "Init"
        ["class"]=>
        string(1) "A"
        ["type"]=>
        string(2) "::"
        ["args"]=>
        array(0) {
        }
      }
    __destruct
    array(1) {
      [0]=>
      array(5) {
        ["function"]=>
        string(10) "__destruct"
        ["class"]=>
        string(1) "A"
        ["object"]=>
        object(A)#2 (0) {
        }
        ["type"]=>
        string(2) "->"
        ["args"]=>
        array(0) {
        }
      }
    }
    __destruct

为什么函数__destruct在这里被调用两次?尤其是第一次。

我想在配置上可能有一些特别的地方,你有什么建议吗?

谢谢。

==================

PS:这个问题不是由"单例设计模式"引起的。同样的问题出现在以下代码中:

<?php
class A
{
    function __construct()
    {   
        echo "__construct'n";
    }
    function __destruct()
    {
        var_dump(debug_backtrace());
        echo "__destruct'n";
    }
}
$a = new A(); 

我终于找到了原因。

这可能是PHP 5.2中的一个错误。X:

如果

zend.ze1_compatibility_mode = On 

那么你可以看到"__destruct" 两次当执行我在我的问题中提供的代码。

此问题已在其他版本中报告:https://bugs.php.net/bug.php?id=29756

在我的测试中不影响PHP 5.3。(PHP 5.3删除了这个设置)

希望这个答案对以后的一些人有帮助:)