首页 > 开发 > linux > 正文

php中为什么先执行后实例化的对象的析构函数

2017-09-11 20:15:17  来源: 网友分享

问题1:问题如题,自己做了测试

class Obj{        public $i;                public function __construct($t){            $this->i = $t;            echo "执行构造函数$this->i";            echo "<br>";        }        public function __destruct(){            echo "执行析构函数$this->i";            echo "<br>";        }    }    $obj1 = new Obj(1);    $obj2 = new Obj(2);执行构造函数1执行构造函数2执行析构函数2执行析构函数1

问题2:
在子类中调用父类的构造方法是否只是对父类进行初始化,是否产生父类的对象?

======================================UPDATE======================================

找到一段理解比较深刻说法:

使用堆还是栈来存储数据是由PHP引擎决定的,PHP开发者不需要关心.
转:
在PHP5的Zend Engine的实现中,所有的值都是在堆上分配空间,并且通过引用计数和垃圾收集来管理.
PHP5的Zend Engine主要使用指向zval结构的指针来操作值,在很多地方甚至通过zval的二级指针来操作.
而在PHP7的Zend Engine实现中,值是通过zval结构本身来操作(非指针).
新的zval结构直接被存放在VM的栈上,HashTable的桶里,以及属性槽里.
这样大大减少了在堆上分配和释放内存的操作,还避免了对简单值的引用计数和垃圾收集.

======================================UPDATE1======================================

找到了具体说明的地方

$p1 = new Person();
对于这个条代码,$p1 是对象名称在栈内存里面new Person()是真正的对象是在堆内存 里面的,具体的请看下图:

这样就解释了为什么先实例化的对象是后释放的

new Person();实际返回的是一个对象的引用,然后引用赋值给$p1,$p1是存储在栈中的变量,是一个指针,指向该对象在堆中分配的实体

这同时也解释了php底层存储变量是有一个hash符号表来维护变量的生命周期的,符号表中存有key=>value键值对,key为变量名称,key指向zval结构体,即value的首地址

解决方案

构造函数和析构函数的执行事实上使用的是一个 结构,由于 Obj(2) 是在后面创建的,因此位于栈顶的位置,按照栈 先进后出 的顺序,销毁时,Obj(2) 就是先被销毁了。