Java的深拷贝和浅拷贝深析

点击进入博客

最近在公司的项目中用到了Java的对象拷贝(Object Copy),准备深入了解一下,试想一下如果我们不对对象拷贝,而是直接将实例赋值给新的实例会如何?

Wheel wheel = new Wheel(20, "朝阳");
Car car = new Car(1, "宝马", wheel);
Car newCar = car;
System.out.println(car);
System.out.println(newCar);

// 输出
objectcopy.Car@7637f22
objectcopy.Car@7637f22

正如以上代码所示,car和newCar都是对堆区内同一个对象的引用,如下图所示,如果我们相对newCar的属性修改,car也会做相应的改动,如果想让newCar生成一个新对象,就要用到Java的对象拷贝。


image

Java的对象拷贝(Object Copy),即把一个对象的属性拷贝到具有相同类型的对象,目的就是为了复用已有的对象的数据,对象拷贝分为深拷贝、浅拷贝和延迟拷贝,其中浅拷贝和深拷贝最为常用,首先了解一下浅拷贝和深拷贝的定义:
浅拷贝:使用一个已知实例对新创建实例的成员变量逐个赋值。
深拷贝: 当一个类的拷贝构造方法,不仅要复制对象的所有非引用成员变量值,还要为引用类型的成员变量创建新的实例,并且初始化为形式参数实例值。
简单的说了浅拷贝只是复制一个对象,传递引用,不能复制实例, 而深拷贝对对象内部的引用均复制。

浅拷贝

浅拷贝是按位拷贝对象(是直接将一个对象的成员值拷贝过来)。如果是基本类型,拷贝的就是基本类型的值;如果属性是引用类型,则会拷贝属性的内存地址。下面的代码是调用Car的clone实现的浅拷贝。

Wheel wheel = new Wheel(20, "朝阳");
Car car = new Car(1, "宝马", wheel);
Car newCar = (Car) car.clone();
System.out.println(car == newCar);
System.out.println(car.getId() == car.getId());
System.out.println(car.getBrand() == car.getBrand());
System.out.println(car.getWheel() == newCar.getWheel());

// 输出
false
true
true
true
image

浅拷贝的实现

浅拷贝的可以通过实现了Clonable接口并重写Object类的clone()方法,然后在方法内部调用super.clone()方法来实现,如下面代码所示

//浅拷贝重写clone方法
@Override
protected Object clone() throws CloneNotSupportedException {
    return (Wheel)super.clone();
}

深拷贝

深拷贝会拷贝所有的属性,并拷贝属性指向的动态分配的内存。当对象和它所引用的对象一起拷贝时即发生深拷贝,深拷贝相对浅拷贝速度较慢并且开销会更大。深拷贝的示例代码如下:

Wheel wheel = new Wheel(20, "朝阳");
Car car = new Car(1, "宝马", wheel);
// 对Car对象进行深拷贝
Car newCar = (Car) BeanUtil.depthClone(car);

System.out.println(car == newCar);
System.out.println(car.getId() == car.getId());
System.out.println(car.getBrand() == car.getBrand());
System.out.println(car.getWheel() == newCar.getWheel());

// 输出
false
true
true
false

从上面结果看与浅拷贝不一样的是wheel没有引用car的wheel对象,而是重新生成了wheel对象,而cat的“id”字段是由基本类型,所以它只是将“id”字段拷贝到newCar的“id”的字段,由此可以看出,深拷贝是完全拷贝原来对象的属性。


image

深拷贝的实现

深拷贝的最常见的实现就是将原对象的每个属性都赋值到新对象的对应的属性中,如下代码重写clone方法

@Override
protected Object clone() throws CloneNotSupportedException {
    Wheel nWheel = (Wheel) wheel.clone();
    return new Car(id, brand, nWheel);
}

@Override
protected Object clone() throws CloneNotSupportedException {
    return new Wheel(diameter, brand);
}

字段较少时,可通过通过逐个字段赋值的方式进行深拷贝,但是如果字段过多,这种方式不免有些臃肿。另一种实现深拷贝的方式就是序列化的方式来实现。如下代码所示,先把srcObj序列化成byte,再用stream去读取byte,再反序列化成cloneObj对象。需要注意的是序列化的类都要实现

public static Object depthClone(Object srcObj) {
        Object cloneObj = null;
        try {
            ByteArrayOutputStream out = new ByteArrayOutputStream();
            ObjectOutputStream oo = new ObjectOutputStream(out);
            oo.writeObject(srcObj);
            ByteArrayInputStream in = new ByteArrayInputStream(out.toByteArray());
            ObjectInputStream oi = new ObjectInputStream(in);
            cloneObj = oi.readObject();
            return cloneObj;
        } catch (Exception ex) {
            return null;
        }
}

详细代码,可以参考代码GitHub

延迟拷贝

延迟拷贝是浅拷贝和深拷贝的一个组合,实际上很少会使用。 当最开始拷贝一个对象时,会使用速度较快的浅拷贝,还会使用一个计数器来记录有多少对象共享这个数据。当程序想要修改原始的对象时,它会决定数据是否被共享(通过检查计数器)并根据需要进行深拷贝。
延迟拷贝从外面看起来就是深拷贝,但是只要有可能它就会利用浅拷贝的速度。当原始对象中的引用不经常改变的时候可以使用延迟拷贝。由于存在计数器,效率下降很高,但只是常量级的开销。而且, 在某些情况下, 循环引用会导致一些问题。

如何选择

如果对象都是基本类型,就可以使用浅拷贝,如果对象有引用类型,这就要看具体的需求了,如果对象是不变的,就可以用浅拷贝,反之就需要深拷贝。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,509评论 6 504
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,806评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,875评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,441评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,488评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,365评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,190评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,062评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,500评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,706评论 3 335
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,834评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,559评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,167评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,779评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,912评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,958评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,779评论 2 354

推荐阅读更多精彩内容