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
}
}

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

目录
相关文章
|
4月前
|
安全 Java 编译器
new出来的对象,不一定在堆上?聊聊Java虚拟机的优化技术:逃逸分析
逃逸分析是一种静态程序分析技术,用于判断对象的可见性与生命周期。它帮助即时编译器优化内存使用、降低同步开销。根据对象是否逃逸出方法或线程,分析结果分为未逃逸、方法逃逸和线程逃逸三种。基于分析结果,编译器可进行同步锁消除、标量替换和栈上分配等优化,从而提升程序性能。尽管逃逸分析计算复杂度较高,但其在热点代码中的应用为Java虚拟机带来了显著的优化效果。
152 4
|
2月前
|
存储 Java Go
【Java】(3)8种基本数据类型的分析、数据类型转换规则、转义字符的列举
牢记类型转换规则在脑海中将编译和运行两个阶段分开,这是两个不同的阶段,不要弄混!
204 2
|
2月前
|
Java Go 开发工具
【Java】(9)抽象类、接口、内部的运用与作用分析,枚举类型的使用
抽象类必须使用abstract修饰符来修饰,抽象方法也必须使用abstract修饰符来修饰,抽象方法不能有方法体。抽象类不能被实例化,无法使用new关键字来调用抽象类的构造器创建抽象类的实例。抽象类可以包含成员变量、方法(普通方法和抽象方法都可以)、构造器、初始化块、内部类(接 口、枚举)5种成分。抽象类的构造器不能用于创建实例,主要是用于被其子类调用。抽象类中不一定包含抽象方法,但是有抽象方法的类必定是抽象类abstract static不能同时修饰一个方法。
217 1
|
8月前
|
监控 Java Unix
6个Java 工具,轻松分析定位 JVM 问题 !
本文介绍了如何使用 JDK 自带工具查看和分析 JVM 的运行情况。通过编写一段测试代码(启动 10 个死循环线程,分配大量内存),结合常用工具如 `jps`、`jinfo`、`jstat`、`jstack`、`jvisualvm` 和 `jcmd` 等,详细展示了 JVM 参数配置、内存使用、线程状态及 GC 情况的监控方法。同时指出了一些常见问题,例如参数设置错误导致的内存异常,并通过实例说明了如何排查和解决。最后附上了官方文档链接,方便进一步学习。
1094 4
|
3月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
|
4月前
|
机器学习/深度学习 安全 Java
Java 大视界 -- Java 大数据在智能金融反洗钱监测与交易异常分析中的应用(224)
本文探讨 Java 大数据在智能金融反洗钱监测与交易异常分析中的应用,介绍其在数据处理、机器学习建模、实战案例及安全隐私等方面的技术方案与挑战,展现 Java 在金融风控中的强大能力。
|
5月前
|
存储 Java 大数据
Java 大视界 -- Java 大数据在智能家居能源消耗模式分析与节能策略制定中的应用(198)
简介:本文探讨Java大数据技术在智能家居能源消耗分析与节能策略中的应用。通过数据采集、存储与智能分析,构建能耗模型,挖掘用电模式,制定设备调度策略,实现节能目标。结合实际案例,展示Java大数据在智能家居节能中的关键作用。
|
6月前
|
数据采集 搜索推荐 算法
Java 大视界 -- Java 大数据在智能教育学习社区用户互动分析与社区活跃度提升中的应用(274)
本文系统阐述 Java 大数据技术在智能教育学习社区中的深度应用,涵盖数据采集架构、核心分析算法、活跃度提升策略及前沿技术探索,为教育数字化转型提供完整技术解决方案。
|
6月前
|
Java 数据库连接 API
互联网大厂校招 JAVA 工程师笔试题解析及常见考点分析
本文深入解析互联网大厂校招Java工程师笔试题,涵盖基础知识(数据类型、流程控制)、面向对象编程(类与对象、继承与多态)、数据结构与算法(数组、链表、排序算法)、异常处理、集合框架、Java 8+新特性(Lambda表达式、Stream API)、多线程与并发、IO与NIO、数据库操作(JDBC、ORM框架MyBatis)及Spring框架基础(IoC、DI、AOP)。通过技术方案讲解与实例演示,助你掌握核心考点,提升解题能力。
252 2
|
7月前
|
人工智能 Java
Java参数传递分析
本文详细探讨了Java中参数传递的机制,明确指出Java采用的是值传递而非引用传递。通过基本数据类型(如int)和引用类型(如Map、自定义对象People)的实例测试,证明方法内部对参数的修改不会影响原始变量。即使在涉及赋值返回的操作中,表面上看似引用传递,实际仍是值传递的结果。文中结合代码示例与执行结果,深入解析了值传递的本质及容易引起混淆的情形,帮助读者准确理解Java参数传递的核心概念。
138 7