PHP对象类型在内存中的分配
对象类型和整型、字符串一样,也是PHP的一种数据类型。都是在程序中用于存储不同类型数据使用,在程序运行时它的每个部分内容都要现加载到内存中在被使用。那么对象类型的数据在内存中是如何分配的呢?
首先,我们了解下内存结构。逻辑上内存大体上被分为四段,分别为栈空间段、堆空间段、初始化数据段、代码段,程序中不同类型数据的声明将会被存放在不同的内存段里面。
1、栈空间段
栈的特点是空间小但被CPU访问的速度快,是用户存放程序中临时创建的变量。由于栈的后进先出特点,所以栈特别方便用来保存和恢复调用现场。从这个意义上讲,我们可以把堆栈看成一个临时数据寄存、交换的内存区。用于存储占用空间长度不变且占用空间小的数据类型的内存段,例如整型(1、100、10000等在内存中占用空间是等长的,占用空间都是32位4个字节)、double、boolean等都可以存储在栈空间段中。
2、堆空间段
堆是用于存放进程运行中被动态分配的内存段,它的大小并不固定,可以动态扩张或缩减。用于存储数据长度可变或占用内存比较大的数据。例如字符串、数组、对象等就存储在堆空间段中。
3、初始化数据段
数据段用来存放可执行文件中已经初始化全局变量,换句话说就是存放程序静态分配的变量。
4、代码段
代码段是用来存放可执行文件的操作指令,也就是说它是可执行程序在内存中的镜像。代码段需要防止在运行时被非法修改,所以只准许读取操作,而不允许写入(修改)操作。例如,程序中的函数就存放在这段内存中。
如图:
程序在运行时,栈内存中的数据是可以直接存取的,而堆内存是不可以直接存取的。对象类型的数据是一种占用空间比较大,而且是占用空间长度不定长的数据类型,所以对象创建完成以后被存放在堆内存中,但对象的引用名称是存放在栈里面的。运行程序时,可以通过对象的引用名称访问对象中的成员。
实例:
<?php /* 声明一个人 类 person */ class Person{ var $name; //成员属性,人的姓名 var $age;//成员属性,人的年龄 var $sex;//成员属性,人的性别 function eat(){ //成员方法,吃饭方式 echo '人在吃饭'; } } //通过person类,实例化三个对象$p1、$p2、$p3 $p1 = new Person(); $p2 = new Person(); $p3 = new Person(); ?>
如图:
从上图可以看到,等号右边是创建的真正对象实例,被存储在堆内存中,而等号左边则是对象的引用,被存储在栈内存段中。
上图一共有三个new Person,所以会在堆内存中开辟出三个独立的空间,产生3个实例对象,每个实例对象都是独立的,独享自己的空间。
每个在堆里面的实例对象是存储属性的,比如说,现在堆里面的实例对象里面都存有姓名、性别和年龄。每个属性又都有一个地址。
$p1=new Person();
等号的左边$p1 是一个引用变量,通过赋值运算符“=”把对象的首地址 赋给“$p1”这个引用变量,所以$p1 是存储对象首地址的变量,$p1 放在栈内存里边,$p1 相当 于一个指针指向堆里面的对象,所以我们可以通过$p1 这个引用变量来操作对象,通常我们也称对象引用为对象。
话外篇:分析下面的代码,你会得出什么理论?
class Person{ public $name = 'test'; } $obj1 = new Person(); $obj1->name = "test1"; echo $obj1->name; //输出:test1 echo '<hr>'; $obj2_0 = clone $obj1; $obj2_0->name = "test2_0"; echo $obj1->name; //输出:test1 echo '<hr>'; echo $obj2_0->name; //输出:test2_0 echo '<hr>'; $obj2 = $obj1; $obj2->name = "test2"; echo $obj1->name; //输出:test2 echo '<hr>'; echo $obj2->name; //输出:test2 echo '<hr>'; $obj3 = new Person(); echo $obj3->name; //输出:test
等号左边 是对象的指针而不是对象本身, 所以obj2和 obj1都指向同一块内存,同一个对象。这一点和OOP语言是一样
object(Person)[2]
public 'name' => string 'test2' (length=5)
object(Person)[2]
public 'name' => string 'test2' (length=5)
可见对象的ID号是一个
如果想得到一个对象的副本,用$obj2 =clone $obj1; 用了clone后会产生一个新对象,分配内存,独立于原来的obj1
$obj2 = $obj1;
$obj2 = &$obj1;
一样的效果,一样的解释?
对于object来说,是一样的。 对于普通的变量是不一样的。
$a = 1;
$b = $a;
$c = &$a;
不一样的 ,具体可以看传值和传引用的区别。
(此篇是根据自己的理解对网上的一篇文章做了一些改动,在此感谢原作者。)