学习C++和Java,一直对深拷贝/浅拷贝迷糊。今天咱就整明白它。
啥是浅拷贝?啥是深拷贝?
看下面的代码。
class Foo {
private Bar myBar;
...
public Foo shallowCopy() {
Foo newFoo = new Foo();
newFoo.myBar = myBar;
return newFoo;
}
public Foo deepCopy() {
Foo newFoo = new Foo();
newFoo.myBar = myBar.clone(); //or new Bar(myBar) or myBar.deepCopy or ...
return newFoo;
}
}
Foo myFoo = new Foo();
Foo sFoo = myFoo.shallowCopy();
Foo dFoo = myFoo.deepCopy();
myFoo.myBar == sFoo.myBar => true
myFoo.myBar.equals(sFoo.myBar) => true
myFoo.myBar == dFoo.myBar => **false**
myFoo.myBar.equals(dFoo.myBar) => true
上面是java代码。很简单,写了个FOO类,里面有两个方法。一个shallowCopy方法就是传说中的浅拷贝;一个deepCopy方法,就是传说中的深拷贝。两个方法都是new了一个FOO对象,然后返回;唯一不同的是,两种方法,在对myBar这个引用处理的时候,有不一样的操作。咱们先来回顾下JVM内存模型。
上图为JVM负责存储的一个笼统的内存模型(更详细的看下探秘系列第一章)。上面我们说到的,myBar这个引用,就是存储在栈里,它指向堆中的一个对象。如果调用的是shallowCopy,那么返回的对象,其中的myBar还是指向堆中原来的那个对象的。如下图(盗了一张dog对象的,一个意思哈)。
再看deepCopy,这里面对myBar处理,使用了clone方法。java中clone,是一个native方法。实际上是在堆中又创建了一个一模一样的对象,然后myBar这个引用,指向了这个新的对象,如下图:
所以,就出现了最开始代码中的结果。如果是同一个对象,只是不同的引用,这样的拷贝就称为浅拷贝;如果是创建了一个新的对象,那就称之为深拷贝。
这就是所谓的深拷贝和浅拷贝。最关键的,就是看在堆里是怎么操作的。是同一个对象,还是又生出了个一模一样的新对象?栈中的引用,只是个表象而已。