PHP开发过程的那些坑(一)——对象拷贝
(原创内容,转载请注明来源,谢谢)
坑:
做单元测试的过程中,想要运用@dataProvider方式分别传两个不同的对象过去。一开始不清楚对象拷贝存在深拷贝和浅拷贝的事情,直接使用等号进行拷贝,导致结果不是想要的结果,调试了很久以后,发现此问题。
举例:
Class Test{
public $a;
public $b;
public function __construct($a, $b){
$this->a = $a;
$this->b = $b;
}
}
$test1 = new Test(1,2);
$test2 = $test1;
$test2->a =2;
此时,$test1和$test2的结果都是2,即$test1的属性$a的值也被改了。不是我想要的结果。
分析:
对象拷贝分为深拷贝和浅拷贝:
深拷贝:赋值时值完全复制,完全的copy,对其中一个作出改变,不会影响另一个。
浅拷贝:赋值时,引用赋值,相当于取了一个别名。对其中一个修改,会影响另一个。使用等号拷贝两个对象,就属于浅拷贝。
改进措施:
1、使用PHP的clone函数
将上述$test2 =$test 1
改为$test2 = clone $test1
后,再对$test2->a
进行复制——$test2->a = 2
,则$test1->a
的值还是1,没有被改动。
2、举一反三,经过查资料,如果使用clone函数,对象中的普通的属性可以实现深拷贝。但是,如果对象的属性中还有对象,即出现类组合的情况,则对此种情况的属性仍是浅拷贝。为了解决此问题,有三种方案:
2.1 方案一
重写类的魔术方法——克隆函数__clone()
在上述Test类中,加一个属性,如下:
Public $obj;//此属性用于存放类的示例
为了深拷贝$obj,则需在上述Test类中,加一个方法,如下:
publicfunction __clone(){
$this->obj= clone $this->obj;
}
但是,在实际工程中,此方法往往不容易实现,因为经常需要涉及的类是框架公共文件、公共接口、其他人或者其他项目的文件等,不易或者没有条件改动里面的源码。因此,下面两个方案更佳。
2.2 方案二
采用序列化和反序列化。
classOther{
public$x;
publicfunction __construct($x){
$this->x = $x;
}
}
class Test{
public $a;
public $b;
public $obj;
public function __construct($a, $b,$obj){
$this->a = $a;
$this->b = $b;
$this->obj = newOther(10);
}
}
$test1 = newTest(1,2);
$test2 = serialize($test1);
$test2 =unserialize($test2);
反序列化后,可以对$test进行操作,不会影响到$test1。
如果安装了PHP的igbinary模块,还可以使用igbinary_serialize和igbinary_unserialize的方式进行序列化和反序列化,对于数据量大的情况下,该方法效率更高,处理速度更快。
2.3 方案三
json_encode和json_decode。
方法和上述类似,可以用此方法将$test1进行转换后赋值给$test2。
——written by linhxx 2017.06.23