Java深浅拷贝分析

简介:

浅拷贝


浅拷贝是按位拷贝对象,它会创建一个新对象,这个对象有着原始对象属性值的一份精确拷贝。如果属性是基本类型,拷贝的就是基本类型的值;如果属性是内存地址(引用类型),拷贝的就是内存地址,因此如果其中一个对象改变了这个地址,就会影响到另一个对象。
简而言之,浅拷贝仅仅复制锁考虑的对象,而不复制它所引用的对象。

实现浅拷贝的步骤是:
1、被复制的类需要实现Cloneable接口(不实现的话在调用clone方法会抛出CloneNotSupportedException异常)改接口为标记接口(不含任何方法)
2、覆盖clone()方法,访问修饰符设为public。方法中调用super.clone()方法得到需要的复制对象,(native为本地方法)

深拷贝


深拷贝还要吧一个对象中所引用的对象也要拷贝一遍。
下面是一个例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
package JavaCoreTechnology;

import java.io.Serializable;

/**
* Created by benjamin on 1/14/16.
*/

public class User implements Serializable, Cloneable {
private static final long serialVersionUID = -3080870074068514966L;

private String userName;
private Dog dog;

public String getUserName() {
return userName;
}

public void setUserName(String userName) {
this.userName = userName;
}

public Dog getDog() {
return dog;
}

public void setDog(Dog dog) {
this.dog = dog;
}

public User clone() throws CloneNotSupportedException {
User user = (User)super.clone();
Dog d = dog.clone(); // 深拷贝必须克隆包含的对象
user.setDog(d);
return user;
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
package JavaCoreTechnology;

import java.io.Serializable;

/**
* Created by benjamin on 1/14/16.
*/

public class Dog implements Serializable, Cloneable {
private static final long serialVersionUID = 8869944898908733919L;

private String dogName;

public String getDogName() {
return dogName;
}

public void setDogName(String dogName) {
this.dogName = dogName;
}

public Dog clone() throws CloneNotSupportedException {
return (Dog)super.clone();
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package JavaCoreTechnology;

/**
* Created by benjamin on 1/14/16.
*/

public class CloneTest {

public static void main(String[] args) throws CloneNotSupportedException {
User user = new User();
Dog dog = new Dog();
dog.setDogName("D1");
user.setUserName("A1");
user.setDog(dog);

User user2 = user.clone();
user2.setUserName("A2");
user2.getDog().setDogName("D2");

System.out.println(user.getUserName()); // A1
System.out.println(user.getDog().getDogName()); // D1

System.out.println(user2.getUserName()); // A2
System.out.println(user2.getDog().getDogName()); // D2
}
}

可以发现上面已经成功的实现了深拷贝。但是试想一下,要是我一个类当中使用到了100个对象,那我也要一个个的去那些对象里面实现clone的方法吗,这显然有些太麻烦了。

为此,我们可以使用序列化对象的方法得到对象的一个拷贝。这就要用到了输入输出流。
修改一下代码,我们再来看一下:

去掉所有实体的Cloneable实现,以及clone方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
package JavaCoreTechnology;

import java.io.Serializable;

/**
* Created by benjamin on 1/14/16.
*/

public class User implements Serializable {
private static final long serialVersionUID = -3080870074068514966L;

private String userName;
private Dog dog;

public String getUserName() {
return userName;
}

public void setUserName(String userName) {
this.userName = userName;
}

public Dog getDog() {
return dog;
}

public void setDog(Dog dog) {
this.dog = dog;
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
package JavaCoreTechnology;

import java.io.Serializable;

/**
* Created by benjamin on 1/14/16.
*/

public class Dog implements Serializable {
private static final long serialVersionUID = 8869944898908733919L;

private String dogName;

public String getDogName() {
return dogName;
}

public void setDogName(String dogName) {
this.dogName = dogName;
}
}

增加一个工具方法用来clone对象:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
package JavaCoreTechnology;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

/**
* Created by benjamin on 1/14/16.
*/

public class ObjCloner {

public static <T> T cloneObj(T obj) {
T cloneVal = null;

try {

// 写入流中
ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(baos);
oos.writeObject(obj);

// 从流中读入对象
ObjectInputStream ois = new ObjectInputStream(new ByteArrayInputStream(baos.toByteArray()));
cloneVal = (T)ois.readObject();

} catch (Exception e) {
e.printStackTrace();
}

return cloneVal;
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package JavaCoreTechnology;

/**
* Created by benjamin on 1/14/16.
*/

public class CloneTest {

public static void main(String[] args) throws CloneNotSupportedException {
User user = new User();
Dog dog = new Dog();
dog.setDogName("D1");
user.setUserName("A1");
user.setDog(dog);

User user2 = ObjCloner.cloneObj(user);
user2.setUserName("A2");
user2.getDog().setDogName("D2");

System.out.println(user.getUserName()); // A1
System.out.println(user.getDog().getDogName()); // D1

System.out.println(user2.getUserName()); // A2
System.out.println(user2.getDog().getDogName()); // D2
}
}

输出结果不变,总得看来,我更推荐大家使用流的方式来进行对象的深拷贝,不知道大家怎么看呢?

目录
相关文章
|
2月前
|
存储 Java
【编程基础知识】 分析学生成绩:用Java二维数组存储与输出
本文介绍如何使用Java二维数组存储和处理多个学生的各科成绩,包括成绩的输入、存储及格式化输出,适合初学者实践Java基础知识。
95 1
|
3月前
|
缓存 JavaScript Java
常见java OOM异常分析排查思路分析
Java虚拟机(JVM)遇到内存不足时会抛出OutOfMemoryError(OOM)异常。常见OOM情况包括:1) **Java堆空间不足**:大量对象未被及时回收或内存泄漏;2) **线程栈空间不足**:递归过深或大量线程创建;3) **方法区溢出**:类信息过多,如CGLib代理类生成过多;4) **本机内存不足**:JNI调用消耗大量内存;5) **GC造成的内存不足**:频繁GC但效果不佳。解决方法包括调整JVM参数(如-Xmx、-Xss)、优化代码及使用高效垃圾回收器。
181 15
常见java OOM异常分析排查思路分析
|
5天前
|
缓存 算法 搜索推荐
Java中的算法优化与复杂度分析
在Java开发中,理解和优化算法的时间复杂度和空间复杂度是提升程序性能的关键。通过合理选择数据结构、避免重复计算、应用分治法等策略,可以显著提高算法效率。在实际开发中,应该根据具体需求和场景,选择合适的优化方法,从而编写出高效、可靠的代码。
19 6
|
4月前
|
Java 程序员
Java 一个 Scanner.nextInt 造成的奇怪问题及分析解决过程
Java 一个 Scanner.nextInt 造成的奇怪问题及分析解决过程
|
28天前
|
监控 算法 Java
jvm-48-java 变更导致压测应用性能下降,如何分析定位原因?
【11月更文挑战第17天】当JVM相关变更导致压测应用性能下降时,可通过检查变更内容(如JVM参数、Java版本、代码变更)、收集性能监控数据(使用JVM监控工具、应用性能监控工具、系统资源监控)、分析垃圾回收情况(GC日志分析、内存泄漏检查)、分析线程和锁(线程状态分析、锁竞争分析)及分析代码执行路径(使用代码性能分析工具、代码审查)等步骤来定位和解决问题。
|
1月前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
69 2
|
1月前
|
Java 关系型数据库 数据库
面向对象设计原则在Java中的实现与案例分析
【10月更文挑战第25天】本文通过Java语言的具体实现和案例分析,详细介绍了面向对象设计的五大核心原则:单一职责原则、开闭原则、里氏替换原则、接口隔离原则和依赖倒置原则。这些原则帮助开发者构建更加灵活、可维护和可扩展的系统,不仅适用于Java,也适用于其他面向对象编程语言。
40 2
|
2月前
|
Java
让星星⭐月亮告诉你,Java synchronized(*.class) synchronized 方法 synchronized(this)分析
本文通过Java代码示例,介绍了`synchronized`关键字在类和实例方法上的使用。总结了三种情况:1) 类级别的锁,多个实例对象在同一时刻只能有一个获取锁;2) 实例方法级别的锁,多个实例对象可以同时执行;3) 同一实例对象的多个线程,同一时刻只能有一个线程执行同步方法。
22 1
|
2月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
60 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
2月前
|
Java
如何从Java字节码角度分析问题|8月更文挑战
如何从Java字节码角度分析问题|8月更文挑战