【面试问题】深拷贝和浅拷贝的区别?

简介: 【1月更文挑战第27天】【面试问题】深拷贝和浅拷贝的区别?

深拷贝和浅拷贝是关于复制对象时引用关系处理的两种不同策略。在理解深拷贝和浅拷贝的概念之前,首先需要了解对象引用和对象内部的数据存储。

对象引用和对象存储:

在 Java 中,对象变量存储的是对象的引用而不是对象本身。当一个对象被赋值给另一个对象变量时,实际上是复制了对象的引用,而不是对象的内容。这意味着两个对象变量可能引用同一个实际对象。、

classPerson {
Stringname;
Person(Stringname) {
this.name=name;
    }
}
publicclassReferenceExample {
publicstaticvoidmain(String[] args) {
Personperson1=newPerson("John");
Personperson2=person1; // person2 引用了 person1 引用的对象person2.name="Doe";
System.out.println(person1.name); // 输出 Doe    }
}

在上述例子中,person2person1 引用同一个 Person 对象,因此对 person2 的修改也会影响到 person1

浅拷贝(Shallow Copy):

浅拷贝是指只复制对象本身,而不复制对象内部的引用类型成员变量所引用的对象。简而言之,它只复制了对象的一层结构,而没有递归地复制对象内部的引用类型。

在 Java 中,可以通过实现 Cloneable 接口并覆盖 clone 方法来实现浅拷贝。

classAddress {
Stringcity;
Address(Stringcity) {
this.city=city;
    }
}
classPersonimplementsCloneable {
Stringname;
Addressaddress;
Person(Stringname, Addressaddress) {
this.name=name;
this.address=address;
    }
@OverrideprotectedObjectclone() throwsCloneNotSupportedException {
returnsuper.clone();
    }
}
publicclassShallowCopyExample {
publicstaticvoidmain(String[] args) throwsCloneNotSupportedException {
Addressaddress=newAddress("New York");
Personperson1=newPerson("John", address);
Personperson2= (Person) person1.clone();
person2.name="Doe";
person2.address.city="Los Angeles";
System.out.println(person1.name); // 输出 JohnSystem.out.println(person1.address.city); // 输出 Los Angeles    }
}

在上述例子中,person2 是通过 person1.clone() 得到的浅拷贝,当修改 person2 的属性时,person1 的属性也会受到影响。这是因为 address 是一个引用类型,浅拷贝只复制了引用,而没有复制引用指向的对象。

深拷贝(Deep Copy):

深拷贝是指在复制对象时,递归地复制对象内部的所有引用类型成员变量所引用的对象。深拷贝会创建一个新的对象,同时复制所有与之相关联的对象。

在 Java 中,实现深拷贝的方式可以是手动编写深拷贝方法,或者使用一些工具类,比如 Apache Commons Lang 库中的 SerializationUtils 或使用序列化和反序列化。

importorg.apache.commons.lang3.SerializationUtils;
classAddress {
Stringcity;
Address(Stringcity) {
this.city=city;
    }
}
classPersonimplementsjava.io.Serializable {
Stringname;
Addressaddress;
Person(Stringname, Addressaddress) {
this.name=name;
this.address=address;
    }
}
publicclassDeepCopyExample {
publicstaticvoidmain(String[] args) {
Addressaddress=newAddress("New York");
Personperson1=newPerson("John", address);
// 使用 SerializationUtils 进行深拷贝Personperson2=SerializationUtils.clone(person1);
person2.name="Doe";
person2.address.city="Los Angeles";
System.out.println(person1.name); // 输出 JohnSystem.out.println(person1.address.city); // 输出 New York    }
}

在上述例子中,SerializationUtils.clone 利用了 Java 对象序列化的特性,将对象序列化为字节流,然后再反序列化为一个新的对象,实现了深拷贝。

相关文章
|
7月前
|
存储 算法 架构师
阿里面试:PS+PO、CMS、G1、ZGC区别在哪?什么是卡表、记忆集、联合表?问懵了,尼恩来一个 图解+秒懂+史上最全的答案
阿里面试:PS+PO、CMS、G1、ZGC区别在哪?什么是卡表、记忆集、联合表?问懵了,尼恩来一个 图解+秒懂+史上最全的答案
|
10月前
|
Java 程序员 调度
Java 高级面试技巧:yield() 与 sleep() 方法的使用场景和区别
本文详细解析了 Java 中 `Thread` 类的 `yield()` 和 `sleep()` 方法,解释了它们的作用、区别及为什么是静态方法。`yield()` 让当前线程释放 CPU 时间片,给其他同等优先级线程运行机会,但不保证暂停;`sleep()` 则让线程进入休眠状态,指定时间后继续执行。两者都是静态方法,因为它们影响线程调度机制而非单一线程行为。这些知识点在面试中常被提及,掌握它们有助于更好地应对多线程编程问题。
435 9
|
10月前
|
安全 Java 程序员
Java面试必问!run() 和 start() 方法到底有啥区别?
在多线程编程中,run和 start方法常常让开发者感到困惑。为什么调用 start 才能启动线程,而直接调用 run只是普通方法调用?这篇文章将通过一个简单的例子,详细解析这两者的区别,帮助你在面试中脱颖而出,理解多线程背后的机制和原理。
506 12
|
10月前
|
编译器 Android开发 开发者
Android经典面试题之Kotlin中Lambda表达式和匿名函数的区别
Lambda表达式和匿名函数都是Kotlin中强大的特性,帮助开发者编写简洁而高效的代码。理解它们的区别和适用场景,有助于选择最合适的方式来解决问题。希望本文的详细讲解和示例能够帮助你在Kotlin开发中更好地运用这些特性。
244 9
|
12月前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
243 14
|
11月前
|
Java
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
今日分享的主题是如何区分&和&&的区别,提高自身面试的能力。主要分为以下四部分。 1、自我面试经历 2、&amp和&amp&amp的不同之处 3、&对&&的不同用回答逻辑解释 4、彩蛋
|
11月前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。
|
存储 Java
【IO面试题 四】、介绍一下Java的序列化与反序列化
Java的序列化与反序列化允许对象通过实现Serializable接口转换成字节序列并存储或传输,之后可以通过ObjectInputStream和ObjectOutputStream的方法将这些字节序列恢复成对象。
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
12月前
|
存储 缓存 算法
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!
本文介绍了多线程环境下的几个关键概念,包括时间片、超线程、上下文切换及其影响因素,以及线程调度的两种方式——抢占式调度和协同式调度。文章还讨论了减少上下文切换次数以提高多线程程序效率的方法,如无锁并发编程、使用CAS算法等,并提出了合理的线程数量配置策略,以平衡CPU利用率和线程切换开销。
面试官:单核 CPU 支持 Java 多线程吗?为什么?被问懵了!