Java的浅拷贝与深拷贝详细解析

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: Java的浅拷贝与深拷贝详细解析

一、认识浅拷贝与深拷贝


对于=赋值,相对于基本数据类型实际上就是直接拷贝它的值,而对于引用数据类型则只是传递这个对象的引用,将原对象的引用实际上还是指向的同一个对象。


浅拷贝:拷贝一个对象时,只对基本数据类型进行拷贝,而对于引用数据类型只是进行了引用的传递,并没有正式的创建一个新的对象。


深拷贝:相对于浅拷贝不同的是,针对于引用数据类型的拷贝是创建了一个新的对象,并且复制其中的成员变量。


引用赋值:


class Person{
    private String name;
    private int age;
    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }
}
public class AnnotationTest {
    @Test
    public void test(){
        Person person = new Person("长路", 18);
        Person person2 = person;
        System.out.println(person);//xyz.changlu.Person@621be5d1
        System.out.println(person2);//xyz.changlu.Person@621be5d1
    }
}



这种方式既不属于浅拷贝也不属于深拷贝就是简单的引用传递。person与person2实例都指向堆中同一个引用地址。

接下来我们来通过例子探讨浅拷贝与深拷贝实现方式。



二、认识clone()方法


首先看一下Object类中的clone()方法:


public class Object {
    //native修饰符说明该方法是一个本地方法
    protected native Object clone() throws CloneNotSupportedException;
}


实现了Cloneable接口即可使用Object的clone()方法。

如何使用这个clone方法呢?需要搭配一个Cloneable接口。


public interface Cloneable {
}


是的该接口没有方法是不是很奇怪。

讲述这两者之间联系:自定义类实现该接口表名当前对象可以clone,实现了该接口后能够改变父类Object类中的clone方法,且能够调用Object的clone方法;若直接调用clone()方法否则会抛出CloneNotSupportedException异常。



三、实现浅拷贝


3.1、clone()方法


对下面的Person进行浅拷贝,利用clone()方法来拷贝:


class Wallet{
    private Integer money = 100;
}
class Person implements Cloneable{
    private String name;
    private int age;
    private Wallet wallet;
    public Person(String name, int age,Wallet wallet) {
        this.name = name;
        this.age = age;
        this.wallet = wallet;
    }
    public Wallet getWallet() {
        return wallet;
    }
    @Override
    public Object clone(){
        Person person = null;
        try {
            person = (Person) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return person;
    }
}
public class AnnotationTest {
    @Test
    public void test() throws CloneNotSupportedException {
        Person person = new Person("长路", 18,new Wallet());
        Person clonePerson = (Person) person.clone();
        //使用clone()获得的对象与原先的引用地址不一样,是在堆中新开辟的
        System.out.println(person == clonePerson);//false
        //查看下其中的引用类型是否直接拷贝了引用,true表示对引用数据类型直接赋引用地址
        System.out.println(person.getWallet() == clonePerson.getWallet());//true
    }
}


首先是Person类实现Cloneable接口。

接着重写了Object的clone()方法,有几个改变点(注意点):

将protected修饰符更改为public,为啥呢,因为使用protected修饰符在不同包中类无法使用,除非该类是Person的子类,方便其他人调用。

不抛出异常,而是直接在方法中catch异常。

关键浅拷贝部分,super.clone()实际上就是调用父类的clone方法(即Object类的),对本实例进行浅拷贝。

总结:使用上面方式的浅拷贝的实例会复制一份新的实例出来,但其中的成员属性内容对于引用类型都是直接拷贝引用而不是重新创建新的对象。(若是想要引用类型属性也重新创建新对象,那么其引用属性也要重写clone()方法,并且在clone好Person之后对其引用属性再重新调用clone()方法)


3.2、System.arraycopy()


该方法属于System类中静态本地方法,是浅拷贝:


public static native void arraycopy(Object src,  int  srcPos,
                                    Object dest, int destPos,
                                    int length);

在jdk中调用该本地方法有:Arrays.copyOf([] original, int newLength, Class<? extends T[]> newType)、ArrayList.clone()


测试程序:通过使用ArrayList来进行测试


class Person{
    private String name = "changlu";
    private int age = 18;
    public void setName(String name) {
        this.name = name;
    }
    public void setAge(int age) {
        this.age = age;
    }
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
public class Test {
    public static void main(String[] args) {
        ArrayList<Person> list = new ArrayList<>();
        list.add(new Person());
        //克隆的ArrayList集合
        ArrayList<Person> clone = (ArrayList<Person>) list.clone();
        clone.get(0).setName("liner");//对该集合中的元素name进行修改看是否改变了原来的对象实例
        //打印原来集合
        System.out.println(list);
    }
}



说明:可以看到对克隆集合中的元素进行修改,同时也改变了原有的集合元素,说明该方法是浅拷贝。



四、实现深拷贝


方式一:使用clone()方法

import org.junit.Test;


class Wallet implements Cloneable{
    private Integer money = 100;
    //1、实现cloneable接口,并重写clone()方法,其中调用Object的clone()方法
    @Override
    public Object clone() {
        Wallet wallet = null;
        try {
            wallet = (Wallet) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return wallet;
    }
    @Override
    public String toString() {
        return "Wallet{" +
                "money=" + money +
                '}';
    }
}
class Person implements Cloneable{
    private String name;
    private int age;
    private Wallet wallet;
    public Person(String name, int age,Wallet wallet) {
        this.name = name;
        this.age = age;
        this.wallet = wallet;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getName() {
        return name;
    }
    public void setWallet(Wallet wallet) {
        this.wallet = wallet;
    }
    public Wallet getWallet() {
        return wallet;
    }
    @Override
    public Object clone(){
        Person person = null;
        try {
            person = (Person) super.clone();
            //2、对person实例中的wallet进行重新拷贝
            person.setWallet((Wallet) person.getWallet().clone());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return person;
    }
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", wallet=" + wallet +
                '}';
    }
}
public class AnnotationTest {
    @Test
    public void test() throws CloneNotSupportedException {
        Person person = new Person("changlu", 18,new Wallet());
        Person clonePerson = (Person) person.clone();
        //使用clone()获得的对象与原先的引用地址不一样,是在堆中新开辟的
        System.out.println(person == clonePerson);//false
        System.out.println(person);
        System.out.println(clonePerson);
        //在Person的clone()方法中对克隆之后的person的属性wallet再次进行clone()
        System.out.println(person.getWallet() == clonePerson.getWallet());//false
        System.out.println(person);
        System.out.println(clonePerson);
        System.out.println(person.getName() == clonePerson.getName());//true
    }
}


实现Person的深拷贝,关键点就是其中的引用数据类型Wallet,那么我们就在Wallet类中实现Cloneable接口,并重写clone()方法即可。

主要改动点就是上面例子中的1、2部分。



其中的String类型并没有进行深拷贝。

弊端说明:很明显我们就能感受到其中的问题所在,若是一个类中有多个自定义的引用类型,那么我们不得一个个类都要去实现Cloneable接口,并重写方法吗?那么我们可以使用反序列化来实现深拷贝;对于jdk中原本定义好的核心类例如String无法进行深拷贝。



方式二:使用反序列化方式

import org.junit.Test;

import java.io.*;


/**
 * @ClassName Test
 * @Author ChangLu
 * @Date 2021/2/21 20:48
 * @Description TODO
 */
class Wallet implements Serializable{
    private static final long serialVersionUID = -6849794470754688710L;
    private Integer money = 100;
    @Override
    public String toString() {
        return "Wallet{" +
                "money=" + money +
                '}';
    }
}
class Person implements Serializable {
    private static final long serialVersionUID = -6849794470754667722L;
    private String name;
    private int age;
    private Wallet wallet;
    public Person(String name, int age,Wallet wallet) {
        this.name = name;
        this.age = age;
        this.wallet = wallet;
    }
    public String getName() {
        return name;
    }
    public void setWallet(Wallet wallet) {
        this.wallet = wallet;
    }
    public Wallet getWallet() {
        return wallet;
    }
    //进行深拷贝
    public Person deepClone() {
        Person person = null;
        ObjectInputStream ois = null;
        ObjectOutputStream oops = null;
        //使用对象输出流进行写操作,写入本身实例
        try{
            //对象输入流将本实例写入
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            oops = new ObjectOutputStream(baos);
            oops.writeObject(this);
            //对象输出流将实例读出
            ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());//读入刚刚写入的baos
            ois = new ObjectInputStream(bais);
            //对象输出流读出实例
            person = (Person) ois.readObject();
        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }finally {
            if(ois != null){
                try {
                    ois.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if(oops != null){
                try {
                    oops.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        return person;
    }
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                ", wallet=" + wallet +
                '}';
    }
}
public class AnnotationTest {
    @Test
    public void test() {
        Person person = new Person("changlu", 18, new Wallet());
        Person clonePerson = person.deepClone();
        //判断克隆的实例是否指向一个引用
        System.out.println(person == clonePerson);//false
        System.out.println(person);
        System.out.println(clonePerson);
        //判断克隆的实例中的引用对象是否指向一个引用
        System.out.println(person.getWallet() == clonePerson.getWallet());//false
        System.out.println(person);
        System.out.println(clonePerson);
        System.out.println(person.getName() == clonePerson.getName());
    }
}


Person与Wallet类都实现了Serializable接口,并且各自类中添加UID。

接着在Person中实现deepClone()方法,该方法中对本身实例进行序列化与反序列化,从而达到深拷贝的效果。



Person类中的String引用类型也进行了深拷贝。

注意:使用反序列化进行深拷贝的对类实现Serializable接口,并添加UID。

相关文章
|
20天前
|
Java 编译器
Java 泛型详细解析
本文将带你详细解析 Java 泛型,了解泛型的原理、常见的使用方法以及泛型的局限性,让你对泛型有更深入的了解。
30 2
Java 泛型详细解析
|
20天前
|
缓存 监控 Java
Java线程池提交任务流程底层源码与源码解析
【11月更文挑战第30天】嘿,各位技术爱好者们,今天咱们来聊聊Java线程池提交任务的底层源码与源码解析。作为一个资深的Java开发者,我相信你一定对线程池并不陌生。线程池作为并发编程中的一大利器,其重要性不言而喻。今天,我将以对话的方式,带你一步步深入线程池的奥秘,从概述到功能点,再到背景和业务点,最后到底层原理和示例,让你对线程池有一个全新的认识。
50 12
|
18天前
|
存储 算法 Java
Java内存管理深度解析####
本文深入探讨了Java虚拟机(JVM)中的内存分配与垃圾回收机制,揭示了其高效管理内存的奥秘。文章首先概述了JVM内存模型,随后详细阐述了堆、栈、方法区等关键区域的作用及管理策略。在垃圾回收部分,重点介绍了标记-清除、复制算法、标记-整理等多种回收算法的工作原理及其适用场景,并通过实际案例分析了不同GC策略对应用性能的影响。对于开发者而言,理解这些原理有助于编写出更加高效、稳定的Java应用程序。 ####
|
18天前
|
存储 监控 算法
Java虚拟机(JVM)垃圾回收机制深度解析与优化策略####
本文旨在深入探讨Java虚拟机(JVM)的垃圾回收机制,揭示其工作原理、常见算法及参数调优方法。通过剖析垃圾回收的生命周期、内存区域划分以及GC日志分析,为开发者提供一套实用的JVM垃圾回收优化指南,助力提升Java应用的性能与稳定性。 ####
|
20天前
|
Java 数据库连接 开发者
Java中的异常处理机制:深入解析与最佳实践####
本文旨在为Java开发者提供一份关于异常处理机制的全面指南,从基础概念到高级技巧,涵盖try-catch结构、自定义异常、异常链分析以及最佳实践策略。不同于传统的摘要概述,本文将以一个实际项目案例为线索,逐步揭示如何高效地管理运行时错误,提升代码的健壮性和可维护性。通过对比常见误区与优化方案,读者将获得编写更加健壮Java应用程序的实用知识。 --- ####
|
23天前
|
数据采集 存储 Web App开发
Java爬虫:深入解析商品详情的利器
在数字化时代,信息处理能力成为企业竞争的关键。本文探讨如何利用Java编写高效、准确的商品详情爬虫,涵盖爬虫技术概述、Java爬虫优势、开发步骤、法律法规遵守及数据处理分析等内容,助力电商领域市场趋势把握与决策支持。
|
22天前
|
存储 缓存 监控
Java中的线程池深度解析####
本文深入探讨了Java并发编程中的核心组件——线程池,从其基本概念、工作原理、核心参数解析到应用场景与最佳实践,全方位剖析了线程池在提升应用性能、资源管理和任务调度方面的重要作用。通过实例演示和性能对比,揭示合理配置线程池对于构建高效Java应用的关键意义。 ####
|
6天前
|
安全 Java API
java如何请求接口然后终止某个线程
通过本文的介绍,您应该能够理解如何在Java中请求接口并根据返回结果终止某个线程。合理使用标志位或 `interrupt`方法可以确保线程的安全终止,而处理好网络请求中的各种异常情况,可以提高程序的稳定性和可靠性。
36 6
|
21天前
|
设计模式 Java 开发者
Java多线程编程的陷阱与解决方案####
本文深入探讨了Java多线程编程中常见的问题及其解决策略。通过分析竞态条件、死锁、活锁等典型场景,并结合代码示例和实用技巧,帮助开发者有效避免这些陷阱,提升并发程序的稳定性和性能。 ####
|
19天前
|
存储 监控 小程序
Java中的线程池优化实践####
本文深入探讨了Java中线程池的工作原理,分析了常见的线程池类型及其适用场景,并通过实际案例展示了如何根据应用需求进行线程池的优化配置。文章首先介绍了线程池的基本概念和核心参数,随后详细阐述了几种常见的线程池实现(如FixedThreadPool、CachedThreadPool、ScheduledThreadPool等)的特点及使用场景。接着,通过一个电商系统订单处理的实际案例,分析了线程池参数设置不当导致的性能问题,并提出了相应的优化策略。最终,总结了线程池优化的最佳实践,旨在帮助开发者更好地利用Java线程池提升应用性能和稳定性。 ####

推荐镜像

更多
下一篇
DataWorks