简单理解浅克隆和深克隆

简介: 简单理解浅克隆和深克隆

定义

浅克隆(Shadow Clone): 拷贝对象时仅仅拷贝对象本身,而不拷贝对象包含的引用指向的对象;对于数据类型是基本数据类型的成员变量,会直接进行值传递,也就是将该属性值复制一份给新的对象。

image.png

深克隆(Deep Clone): 将原型对象中所有类型,无论是值类型还是引用类型,都复制一份给克隆对象。

image.png

浅克隆的实现:

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

@Data
@AllArgsConstructor
@NoArgsConstructor
public class Monkey implements Cloneable {
    private String name;
    private int age;

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }


    public static void main(String[] args) throws CloneNotSupportedException {
        Monkey monkey1 = new Monkey("大明", 10);
        Monkey monkey2 = (Monkey) monkey1.clone();

        System.out.println(monkey1);
        System.out.println(monkey2);
    }
}

输出结果:
image.png

浅克隆存在弊端: 由于基本数据类型是直接存储在栈内存中的,而引用数据类型,则仅仅是把地址存储在栈内存中,真正的数据是存储在堆内存中的,赋值操作时,仅仅把地址进行了赋值。这样就会存在两个对象的成员变量同时指向一个实例,这种情况下,在对一个对象中的该成员变量进行修改会影响到另一个对象的该成员变量值。
例如:

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

@Data
@AllArgsConstructor
@NoArgsConstructor
public class Park implements Cloneable {

    private String name;
    private Monkey monkey;


    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }


    public static void main(String[] args) throws CloneNotSupportedException {
        Monkey monkey1 = new Monkey("大明", 10);
        Park park1 = new Park("金山动物园", monkey1);
        Park park2 = (Park) park1.clone();

        System.out.println("修改引用变量前:>>>>park1>>>>" + park1);
        System.out.println("修改引用变量前:>>>>park2>>>>" + park2);

        Monkey monkey2 = park2.getMonkey();
        monkey2.setName("二明");
        monkey2.setAge(9);

        System.out.println("修改引用变量后:>>>>park1>>>>" + park1);
        System.out.println("修改引用变量后:>>>>park2>>>>" + park2);
    }
}

输出:
image.png

这里看到,修改了park2中的monkey2变量,导致park1中的monkey1变量也发生了变化。所以这里引入了深克隆。即要克隆类的基本数据类型,以及所有非基本数据类型的属性。

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

@Data
@AllArgsConstructor
@NoArgsConstructor
public class Park implements Cloneable {

    private String name;
    private Monkey monkey;


    @Override
    protected Object clone() throws CloneNotSupportedException {
        /**
         * 主要是这里
         */
        Park park = (Park)super.clone();
        park.setMonkey((Monkey) park.getMonkey().clone());
        return park;
    }


    public static void main(String[] args) throws CloneNotSupportedException {
        Monkey monkey1 = new Monkey("大明", 10);
        Park park1 = new Park("金山动物园", monkey1);
        Park park2 = (Park) park1.clone();

        System.out.println("修改引用变量前:>>>>park1>>>>" + park1);
        System.out.println("修改引用变量前:>>>>park2>>>>" + park2);

        Monkey monkey2 = park2.getMonkey();
        monkey2.setName("二明");
        monkey2.setAge(9);

        System.out.println("修改引用变量后:>>>>park1>>>>" + park1);
        System.out.println("修改引用变量后:>>>>park2>>>>" + park2);
    }
}

输出结果:
image.png

这里简单的示例帮助理解深克隆和浅克隆。

深克隆方式

1、实现Cloneable接口

2、使用JDK自带的字节流实现深克隆

3、使用第三方工具类实现深克隆,比如Apache Commons Lang

4、使用Json工具实现深克隆,比如Gson、FastJSON,使用JSON.pasre和JSON.stringify。

相关文章
|
5月前
|
存储
类与对象\深复制与浅复制
类与对象\深复制与浅复制
24 0
|
设计模式 Java 程序员
谈谈你对深克隆和浅克隆的理解
一个工作了5年的程序员私信我说,前几天去阿里面试被问到这样一个这样的面试题,说谈谈你对深克隆和浅克隆的理解。他回答说深克隆是克隆值,浅克隆是克隆引用,当时他只说了这样一句话,回答完以后,他看到面试官的表情很诧异,面试也没有继续深入追问了。小伙伴们,如果是你来回答,你也会这样回回答吗?
84 0
|
6月前
|
存储 设计模式
浅克隆和深克隆的详细教程~
浅克隆和深克隆的详细教程~
|
6月前
|
消息中间件 Kubernetes NoSQL
构造函数、深拷贝、浅拷贝
构造函数、深拷贝、浅拷贝
|
设计模式 Java API
一文读懂深克隆与浅克隆的关系
在Java提供的API中,不需要手动创建抽象原型接口,因为Java已经内置了Cloneable抽象原型接口,自定义的类型只需实现该接口并重写Object.clone()方法即可完成本类的复制。 通过查看JDK的源码可以发现,其实Cloneable是一个空接口。Java之所以提供Cloneable接口,只是为了在运行时通知Java虚拟机可以安全地在该类上使用clone()方法。而如果该类没有实现 Cloneable接口,则调用clone()方法会抛出 CloneNotSupportedException异常。 一般情况下,如果使用clone()方法,则需满足以下条件。
68 0
|
C++
C++类与对象中深拷贝与浅拷贝
C++类与对象中深拷贝与浅拷贝
68 0
|
设计模式 C++
再会原型模式——深复制VS浅复制
再会原型模式——深复制VS浅复制
104 0
再会原型模式——深复制VS浅复制
|
Java 测试技术 uml
好似你,饼印咁:原型模式、浅拷贝与深拷贝
用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。大概意思就是:允许一个对象再创建另外一个可定制的对象,根本无需知道对象创建的细节。
134 0
好似你,饼印咁:原型模式、浅拷贝与深拷贝
一篇文章带你了解cloneable接口、浅拷贝、深拷贝
我们都知道想要实现拷贝需要实现Cloneable接口并在类中实现clone()方法,不过比较神奇的是,clone()方法并不是Cloneable接口中的方法。 Cloneable接口是一个空接口,里面没有任何内容
|
Java API
Java对象的深克隆与浅克隆(对象复制)(上)
Java对象的深克隆与浅克隆(对象复制)(上)
130 0
Java对象的深克隆与浅克隆(对象复制)(上)