深拷贝:赋值时值彻底复制,彻底的copy,对其中一个做出改变,不会影响另外一个php
浅拷贝:赋值时,引用赋值,至关于取了一个别名。对其中一个修改,会影响另外一个html
对于PHP而言,= 赋值时,普通对象是深拷贝,但对对象来讲,是浅拷贝,即引用赋值。当对象做为参数传递时,不管参数前是否有&引用符号,都将被看作是赋值引用。python
对于python而言,状况可能会有点小复杂,由于python一切皆为对象,因此python的普通赋值、深拷贝和浅拷贝之间都是有细微区别的。编程
在php5中,对象的 = 赋值和传递都是引用。要想实现拷贝副本,php提供了clone函数实现。clone彻底copy了一份副本。可是clone时,咱们可能不但愿copy源对象的全部内容,那咱们能够利用__clone来操做。
请看以下代码段:函数
<?php //普通对象赋值,深拷贝,彻底值复制 $m = 1; $n = $m; $n = 2; echo $m;//值复制,对新对象的改变不会对m做出改变,输出 1.深拷贝 echo PHP_EOL; /*==================*/ //对象赋值,浅拷贝,引用赋值 class Test{ public $a=1; } $m = new Test(); $n = $m;//引用赋值 $m->a = 2;//修改m,n也随之改变 echo $n->a;//输出2,浅拷贝 echo PHP_EOL; ?>
因为对象的赋值时引用,要想实现值复制,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,随新对象改变 ?>
要想实现对象真正的深拷贝,有如下两种方法:this
一、利用序列化反序列化实现spa
<?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,不随新对象改变,仍是保持了原来的属性,能够看到,序列化和反序列化能够实现对象的深拷贝 ?>
二、写clone函数.net
<?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,不随新对象改变,仍是保持了原来的属性 ?>
“对一个对象进行浅拷贝实际上是新建立了一个类型和原来对象同样,可是内容是原来对象元素的引用。换句话说,这个拷贝的对象自己是新的,可是它的内容不是”,摘自《Python核心编程》。code
这是我我的对python下浅拷贝和深拷贝的理解:
赋值:简单地拷贝对象的引用,两个对象的id相同。
浅拷贝:建立一个新的组合对象,这个新对象与原对象共享内存中的子对象。
深拷贝:建立一个新的组合对象,同时递归地拷贝全部子对象,新的组合对象与原对象没有任何关联。虽然实际上会共享不可变的子对象,但不影响它们的相互独立性。
浅拷贝和深拷贝的不一样仅仅是对组合对象来讲,所谓的组合对象就是包含了其它对象的可变对象,如列表,类实例。而对于数字、字符串以及其它“原子”类型,没有拷贝一说,产生的都是原对象的引用。
下面的代码但愿能对你有进一步的帮助;
#! /usr/bin/python # -*- coding:UTF-8 -*- import copy # 浅拷贝 a = [1, "a", 3, [4, 5, 6], [[7, 8, 9]]] b = a c = list(a) d = copy.deepcopy(a) print "原地址&&&" print id(a) print "赋值地址&&&" print id(b) print "浅拷贝地址&&&" print id(c) print "深拷贝地址&&&" print id(d) print "赋值地址###" for i, j in zip(a, b): print id(i), id(j) print "浅拷贝地址###" for i, j in zip(a, c): print id(i), id(j) print "深拷贝地址###" for i, j in zip(a, d): print id(i), id(j) print "######" a[0] = 2 a[3][0] = 14 print "原值变化为 %d, %d" % (a[0], a[3][0]) print "*******" print "赋值变化" print b[0], b[3][0] print "浅拷贝变化" print c[0], c[3][0] print "深拷贝变化" print d[0], d[3][0] print "**##$$" print a
输出以下:
http://www.cnblogs.com/taijun...
http://blog.csdn.net/u0115085...
http://www.cnblogs.com/zxlove...