当前位置:编程学习 > php >>

深入讨论PHP5对象复制技术

 

 

此文将由浅入深的讨论PHP5的对象复制技术  原创文章 请尊重版权  有错误或则不当之处还希望能够指出来

对象复制的由来

为什么对象会有“复制”这个概念,这与PHP5中对象的传值方式是密切相关的,让我们看看下面这段简单的代码

 

PHP代码

 

 

  1. /** 
  2.  * 电视机类 
  3.  */  
  4. class Television   
  5. {  
  6.     /** 
  7.      * 屏幕高度 
  8.      */  
  9.     protected $_screenLength = 300;  
  10.       
  11.     /** 
  12.      * 屏幕宽度 
  13.      */  
  14.     protected $_screenHight  = 200;  
  15.       
  16.     /** 
  17.      * 电视机外观颜色 
  18.      */  
  19.     protected $_color        = 'black';  
  20.       
  21.     /** 
  22.      * 返回电视外观颜色 
  23.      */  
  24.     public function getColor()  
  25.     {  
  26.         return $this->_color;  
  27.     }  
  28.       
  29.     /** 
  30.      * 设置电视机外观颜色 
  31.      */  
  32.     public function setColor($color)  
  33.     {  
  34.         $this->_color = (string)$color;  
  35.         return $this;  
  36.     }  
  37. }  
  38.   
  39. $tv1 = new Television();  
  40. $tv2 = $tv1;  

 

这段代码定义了一个电视机的类 Television , $tv1为一个电视机的实例,然后我们按照普通的变量赋值方式将$tv1的值赋给$t2。那么现在我们拥有两台电视机$tv1和$tv2了,真的是这样的吗?我们来测试一下。

 

PHP代码

 

 

  1. echo 'color of tv1 is: ' . $tv1->getColor();//tv1的颜色是black  
  2. echo '<br>';  
  3. echo 'color of tv2 is: ' . $tv2->getColor();//tv2的颜色是black  
  4. echo '<br>';  
  5.   
  6. //把tv2涂成白色  
  7. $tv2->setColor('white');  
  8.   
  9. echo 'color of tv2 is: ' . $tv2->getColor();//tv2的颜色是white  
  10. echo '<br>';  
  11. echo 'color of tv1 is: ' . $tv1->getColor();//tv1的颜色是white  

 

 首先我们看到tv1和tv2的颜色都是black,现在我们希望tv2换个颜色,所以我们将它的颜色设置成了white,我们再看看tv2的颜色,确实成为了white,似乎满足了我们的要求,可是并没有想象中的那么顺利,当我们接着看tv1的颜色的时候,我们发现tv1也由black边成了white。我们并没有重新设置tv1的颜色,为什么tv1会重black变成white呢?这是因为PHP5中对象的赋值和传值都是以“引用”的方式。PHP5使用了Zend引擎II,对象被储存于独立的结构Object Store中,而不像其它一般变量那样储存于Zval中(在PHP4中对象和一般变量一样存储于Zval)。在Zval中仅存储对象的指针而不是内容(value)。当我们复制一个对象或者将一个对象当作参数传递给一个函数时,我们不需要复制数据。仅仅保持相同的对象指针并由另一个zval通知现在这个特定的对象指向的Object Store。由于对象本身位于Object Store,我们对它所作的任何改变将影响到所有持有该对象指针的zval结构----表现在程序中就是目标对象的任何改变都会影响到源对象。.这使PHP对象看起来就像总是通过引用(reference)来传递。所以以上的tv2和tv1其实是指向同一个电视机实例,我们对tv1或则tv2所做的操作其实都是针对这同一个实例。因此我们的“复制”失败了。看来直接变量赋值的方式并不能拷贝对象,为此PHP5提供了一个专门用于复制对象的操作,也就是 clone 。这就是对象复制的由来。

用clone(克隆)来复制对象

我们现在使用PHP5的clone语言结构来复制对象,代码如下:

PHP代码

 

  1. $tv1 = new Television();  
  2. $tv2 = clone $tv1;  
  3.   
  4. echo 'color of tv1 is: ' . $tv1->getColor();//tv1的颜色是black  
  5. echo '<br>';  
  6. echo 'color of tv2 is: ' . $tv2->getColor();//tv2的颜色是black  
  7. echo '<br>';  
  8.   
  9. //把tv2换成涂成白色  
  10. $tv2->setColor('white');  
  11.   
  12. echo 'color of tv2 is: ' . $tv2->getColor();//tv2的颜色是white  
  13. echo '<br>';  
  14. echo 'color of tv1 is: ' . $tv1->getColor();//tv1的颜色是black  

 

这段代码的第2行,我们用clone关键字复制tv1,现在我们就拥有了一份真正的tv1的拷贝tv2,我们还是按照之前的方法来检测复制是否成功。我们可以看到,我们将tv2的颜色换成了white,tv1的颜色还是black,这样我们的复制操作就成功了。

__clone魔术方法

    现在我们考虑到这样一个情况,每一台电视机应该都有自己的编号,这个编号如同我们的身份证号码一样应该是唯一的,所以当我们在复制一台电视机的时候,我们不希望这个编号也被复制过来,以免造成一些麻烦。我们想到的一个策略是将赋值出来的电视机的编号清空,然后再按照需求来重新分配编号。

    那么__clone魔术方法就是专门用来解决这样的问题,__clone魔术方法会在对象被复制( 也就是clone操作)的时候被触发。我们修改了电视机类Television的代码,添加了编号属性和__clone方法,代码如下。

PHP代码

 

 

  1. /** 
  2.  * 电视机类 
  3.  */  
  4. class Television   
  5. {  
  6.       补充:Web开发 , php ,
CopyRight © 2012 站长网 编程知识问答 www.zzzyk.com All Rights Reserved
部份技术文章来自网络,