PHP开发过程的那些坑(一) ——对象拷贝

时间:2022-05-03
本文章向大家介绍PHP开发过程的那些坑(一) ——对象拷贝,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

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