PHP中深拷貝和淺拷貝的區(qū)別:1、深拷貝賦值時(shí)是完全復(fù)制,而淺拷貝只是引用賦值,相當(dāng)于取了一個(gè)別名;2、深拷貝若對(duì)其中一個(gè)做出改變不會(huì)影響另一個(gè),而淺拷貝對(duì)其中一個(gè)進(jìn)行修改會(huì)影響另一個(gè)。
本教程操作環(huán)境:windows10系統(tǒng)、PHP7.1版、DELL G3電腦
php中深拷貝和淺拷貝的區(qū)別是什么
先說一下深拷貝和淺拷貝通俗理解
深拷貝:賦值時(shí)值完全復(fù)制,完全的copy,對(duì)其中一個(gè)作出改變,不會(huì)影響另一個(gè)
淺拷貝:賦值時(shí),引用賦值,相當(dāng)于取了一個(gè)別名。對(duì)其中一個(gè)修改,會(huì)影響另一個(gè)
PHP中, = 賦值時(shí),普通對(duì)象是深拷貝,但對(duì)對(duì)象來說,是淺拷貝。也就是說,對(duì)象的賦值是引用賦值。(對(duì)象作為參數(shù)傳遞時(shí),也是引用傳遞,無論函數(shù)定義時(shí)參數(shù)前面是否有&符號(hào))
php4中,對(duì)象的 = 賦值是實(shí)現(xiàn)一份副本,這樣存在很多問題,在不知不覺中我們可能會(huì)拷貝很多份副本。
php5中,對(duì)象的 = 賦值和傳遞都是引用。要想實(shí)現(xiàn)拷貝副本,php提供了clone函數(shù)實(shí)現(xiàn)。
clone完全copy了一份副本。但是clone時(shí),我們可能不希望copy源對(duì)象的所有內(nèi)容,那我們可以利用__clone來操作。
在__clone()中,我們可以進(jìn)行一些操作。注意,這些操作,也就是__clone函數(shù)是作用于拷貝的副本對(duì)象上的
<?php //普通對(duì)象賦值,深拷貝,完全值復(fù)制 $m = 1; $n = $m; $n = 2; echo $m;//值復(fù)制,對(duì)新對(duì)象的改變不會(huì)對(duì)m作出改變,輸出 1.深拷貝 echo PHP_EOL; /*==================*/ //對(duì)象賦值,淺拷貝,引用賦值 class Test{ public $a=1; } $m = new Test(); $n = $m;//引用賦值 $m->a = 2;//修改m,n也隨之改變 echo $n->a;//輸出2,淺拷貝 echo PHP_EOL; ?>
由于對(duì)象的賦值時(shí)引用,要想實(shí)現(xiàn)值復(fù)制,php提供了clone函數(shù)來實(shí)現(xiàn)復(fù)制對(duì)象。
但是clone函數(shù)存在這么一個(gè)問題,克隆對(duì)象時(shí),原對(duì)象的普通屬性能值復(fù)制,但是源對(duì)象的對(duì)象屬性賦值時(shí)還是引用賦值,淺拷貝。
<?php class Test{ public $a=1; } class TestOne{ public $b=1; public $obj; //包含了一個(gè)對(duì)象屬性,clone時(shí),它會(huì)是淺拷貝 public function __construct(){ $this->obj = new Test(); } } $m = new TestOne(); $n = $m;//這是完全的淺拷貝,無論普通屬性還是對(duì)象屬性 $p = clone $m; //普通屬性實(shí)現(xiàn)了深拷貝,改變普通屬性b,不會(huì)對(duì)源對(duì)象有影響 $p->b = 2; echo $m->b;//輸出原來的1 echo PHP_EOL; //對(duì)象屬性是淺拷貝,改變對(duì)象屬性中的a,源對(duì)象m中的對(duì)象屬性中a也改變 $p->obj->a = 3; echo $m->obj->a;//輸出3,隨新對(duì)象改變 ?>
要想實(shí)現(xiàn)對(duì)象真正的深拷貝,有下面兩種方法:
寫clone函數(shù):如下
<?php class Test{ public $a=1; } class TestOne{ public $b=1; public $obj; //包含了一個(gè)對(duì)象屬性,clone時(shí),它會(huì)是淺拷貝 public function __construct(){ $this->obj = new Test(); } //方法一:重寫clone函數(shù) public function __clone(){ $this->obj = clone $this->obj; } } $m = new TestOne(); $n = clone $m; $n->b = 2; echo $m->b;//輸出原來的1 echo PHP_EOL; //可以看到,普通屬性實(shí)現(xiàn)了深拷貝,改變普通屬性b,不會(huì)對(duì)源對(duì)象有影響 //由于改寫了clone函數(shù),現(xiàn)在對(duì)象屬性也實(shí)現(xiàn)了真正的深拷貝,對(duì)新對(duì)象的改變,不會(huì)影響源對(duì)象 $n->obj->a = 3; echo $m->obj->a;//輸出1,不隨新對(duì)象改變,還是保持了原來的屬性 ?>
改寫__clone()函數(shù)不太方便,而且你得在每個(gè)類中把這個(gè)類里面的對(duì)象屬性都在__clone()中 一一 clone
第二種方法,利用序列化反序列化實(shí)現(xiàn),這種方法實(shí)現(xiàn)對(duì)象的深拷貝簡(jiǎn)單,不需要修改類
<?php class Test{ public $a=1; } class TestOne{ public $b=1; public $obj; //包含了一個(gè)對(duì)象屬性,clone時(shí),它會(huì)是淺拷貝 public function __construct(){ $this->obj = new Test(); } } $m = new TestOne(); //方法二,序列化反序列化實(shí)現(xiàn)對(duì)象深拷貝 $n = serialize($m); $n = unserialize($n); $n->b = 2; echo $m->b;//輸出原來的1 echo PHP_EOL; //可以看到,普通屬性實(shí)現(xiàn)了深拷貝,改變普通屬性b,不會(huì)對(duì)源對(duì)象有影響 $n->obj->a = 3; echo $m->obj->a;//輸出1,不隨新對(duì)象改變,還是保持了原來的屬性,可以看到,序列化和反序列化可以實(shí)現(xiàn)對(duì)象的深拷貝 ?>
還有第三種方法,其實(shí)和第二種類似,json_encode之后再json_decode,實(shí)現(xiàn)賦值
推薦學(xué)習(xí):《PHP視頻教程》