如何复制PHP对象的且更改值互不影响

今天在项目中有一个需求,需要复制一份DB类查询出来的结果作为原数据和新数据进行比较。

一开始我直接复制了一份:$walletInfoCopy = $walleltInfo;

但是问题是,如果$walletInfo中的值更改,那么$walletInfoCopy中的值也会更改,后来发现只要实现PHP对象的完全深拷贝就可以实现这个需求。

先说一下深拷贝和浅拷贝通俗理解

  • 深拷贝:赋值时值完全复制,完全的copy,对其中一个作出改变,不会影响另一个
  • 浅拷贝:赋值时,引用赋值,相当于取了一个别名。对其中一个修改,会影响另一个

PHP中, = 赋值时,普通对象是深拷贝,但对对象来说,是浅拷贝。也就是说,对象的赋值是引用赋值。(对象作为参数传递时,也是引用传递,无论函数定义时参数前面是否有&符号)

普通数据类型一般是深拷贝,而对象的深拷贝分为三种,浅拷贝、部分深拷贝、完全深拷贝,下面一一介绍:

1、普通数据类型的深拷贝 ‘=’

<?php

//普通对象赋值,深拷贝,完全值复制
$m = 1;
$n = $m;
$n = 2;
echo $m;//值复制,对新对象的改变不会对m作出改变,输出 1.深拷贝
echo PHP_EOL;
?>

2、对象的浅拷贝 ‘=’

//对象赋值,浅拷贝,引用赋值

class Test{

    public $a=1;

}

$m = new Test();

$n = $m;//引用赋值

$m->a = 2;//修改m,n也随之改变

echo $n->a;//输出2,浅拷贝

echo PHP_EOL;

?>

3、clone实现对象的部分深拷贝

由于对象的赋值时引用,要想实现值复制,php提供了clone函数来实现复制对象。

但是clone函数存在这么一个问题,克隆对象时,原对象的普通属性能值复制,但是源对象的对象属性赋值时还是引用赋值,浅拷贝。

<?php

class Test{

    public $a=1;

}



class TestOne{

    public $b=1;

    public $obj;

    //包含了一个对象属性,clone时,它会是浅拷贝

    public function __construct(){

        $this->obj = new Test();

    }

}

$m = new TestOne();

$n = $m;//这是完全的浅拷贝,无论普通属性还是对象属性



$p = clone $m;



//普通属性实现了深拷贝,改变普通属性b,不会对源对象有影响

$p->b = 2;

echo $m->b;//输出原来的1

echo PHP_EOL;



//对象属性是浅拷贝,改变对象属性中的a,源对象m中的对象属性中a也改变



$p->obj->a = 3;

echo $m->obj->a;//输出3,随新对象改变

?>

要想实现对象真正的深拷贝,有下面两种方法:

写clone函数:如下

<?php

class Test{

    public $a=1;

}



class TestOne{

    public $b=1;

    public $obj;

    //包含了一个对象属性,clone时,它会是浅拷贝

    public function __construct(){

        $this->obj = new Test();

    }

     

    //方法一:重写clone函数

    public function __clone(){

        $this->obj = clone $this->obj;

    }

}



$m = new TestOne();

$n = clone $m;



$n->b = 2;

echo $m->b;//输出原来的1

echo PHP_EOL;

//可以看到,普通属性实现了深拷贝,改变普通属性b,不会对源对象有影响



//由于改写了clone函数,现在对象属性也实现了真正的深拷贝,对新对象的改变,不会影响源对象

$n->obj->a = 3;

echo $m->obj->a;//输出1,不随新对象改变,还是保持了原来的属性



?>


第二种方法,利用序列化反序列化实现,这种方法实现对象的深拷贝简单,不需要修改类  改写__clone()函数不太方便,而且你得在每个类中把这个类里面的对象属性都在__clone()中 一一 clone

<?php

class Test{

    public $a=1;

}



class TestOne{

    public $b=1;

    public $obj;

    //包含了一个对象属性,clone时,它会是浅拷贝

    public function __construct(){

        $this->obj = new Test();

    }

     

}



$m = new TestOne();

//方法二,序列化反序列化实现对象深拷贝

$n = serialize($m);

$n = unserialize($n);



$n->b = 2;

echo $m->b;//输出原来的1

echo PHP_EOL;

//可以看到,普通属性实现了深拷贝,改变普通属性b,不会对源对象有影响





$n->obj->a = 3;

echo $m->obj->a;//输出1,不随新对象改变,还是保持了原来的属性,可以看到,序列化和反序列化可以实现对象的深拷贝



?>

 还有第三种方法,其实和第二种类似,json_encode之后再json_decode,实现赋值
 

来源:https://www.cnblogs.com/taijun/p/4208008.html

发布了10 篇原创文章 · 获赞 5 · 访问量 7382

猜你喜欢

转载自blog.csdn.net/qq_41566366/article/details/83749438