PHP的深拷贝与浅拷贝

什么是拷贝

拷贝是由英文copy的音译词,拼音是kǎo bèi。copy意为复制、摹本。拷贝有四种解释:一指一个计算机系统的DOS命令,意为“复制”,是一个新名词,广泛地应用于IT的各个领域;二指由底片复制出来供放映电影用的胶片,如巴金的《随想录·再谈<望乡>》中说到:“我们最初就是根据这个拷贝放映的。”;三指复写,如拷贝纸、拷贝笔等;四指一个流行语,指复印、照抄、抄袭等意思,贬义词。

今天我们要说的拷贝就是指的第一种解释,计算中的拷贝。这里的拷贝的内容就是指存储在计算机内存中的二进制数据。

什么是深浅拷贝

1.深拷贝
深拷贝就是在内存中开辟一块新的空间,将复制的对象存储在这个新的空间中。如下:

1
2
$a = 1;
$b = $a;// 创建了一个新的变量$b(开辟了一块新的空间),复制的对象就是变量$a的内容。

2.浅拷贝
浅拷贝就是共用同一片内存空间,使用的是引用传值。如下:

1
2
3
$a = 0;
$b = &$b;// 创建一个新的变量,变量指向$a的内存空间。
$b = 1;// 此时$a,$b使用的就是同一片内存,因此值就是1。

变量拷贝

可参考深拷贝和浅拷贝中的演示示例

对象拷贝

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

php4中,对象的 = 赋值是实现一份副本,这样存在很多问题,在不知不觉中我们可能会拷贝很多份副本。

php5中,对象的 = 赋值和传递都是引用。要想实现拷贝副本,php提供了clone函数实现。
clone完全copy了一份副本。但是clone时,我们可能不希望copy源对象的所有内容,那我们可以利用__clone来操作。
1.浅拷贝

1
2
3
4
5
6
7
8
9
10
11
12
13
14
<?php
declare (strict_types = 1);

class Test
{
public $name = '张三';
public function __construct()
{
}
}

$test = new Test();
$test->name = '李思';
var_dump($test->name);

2.深拷贝

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<?php
declare (strict_types = 1);

class Test
{
public $name = '张三';
public function __construct()
{
}
}

$test = new Test();
var_dump($test);

$n = clone $test;
$n->name = '李思';
var_dump($n->name);

clone()中,我们可以进行一些操作。注意,这些操作,也就是clone函数是作用于拷贝的副本对象上的
但是clone函数存在这么一个问题,克隆对象时,原对象的普通属性能值复制,但是源对象的对象属性赋值时还是引用赋值,浅拷贝。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
<?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,随新对象改变
?>

要想实现对象真正的深拷贝,有如下3种方法:
1.使用__clone方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
<?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,不随新对象改变,还是保持了原来的属性

?>

2.使用serialize和unserialize

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
<?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,不随新对象改变,还是保持了原来的属性,可以看到,序列化和反序列化可以实现对象的深拷贝

?>

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