简介

拷贝对象是java中经常会遇到的问题。java中存在两种类型,基础类型和引用类型。

java的赋值都是传值的,对于基础类型来说,会拷贝具体的内容,但是对于引用对象来说,存储的这个值只是指向实际对象的地址,拷贝也只会拷贝引用地址。

因为引用对象的存在,所以经常会出现和预期不一样的情况。

本文将会深入的探讨一下在拷贝对象中会出现的浅拷贝和深拷贝的情况。

拷贝接口

java中所有的对象都是继承自java.lang.Object。Object对象中提供了一个clone方法,来供我们对java对象进行拷贝。

    protected native Object clone() throws CloneNotSupportedException;

这个clone方法是native的,所以不需要我们来实现,但是注意clone方法还是protected,这意味着clone方法只能在java.lang包或者其子类可见。

如果我们想要在一个程序中调用某个对象的clone方法则是不可以的。因为clone方法是定义在Object中的,该对象并没有对外可见的clone方法。

JDK的建议是让我们去实现接口Cloneable,实现了这个接口就表示这个对象可以调用Object的clone方法。

注意,即使你实现了Cloneable接口,还是无法在外部程序中调用该对象的clone方法:

public interface Cloneable {
}

因为Cloneable是空的,明没有强制要你去实现clone方法。

这是JDK在设计上的问题,导致clone方法并不像预期那么好用。

首先clone只是对象的拷贝,它只是简单的拷贝对象,而不会去执行对象的构造函数。

其次clone会导致浅拷贝的问题。

使用clone导致的浅拷贝

我们举个clone产生的浅拷贝的例子,我们定义一个对象中的对象,然后尝试拷贝:

@Data
public class Address implements Cloneable{
private String name; //不是好的方式
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone(); }
}
@Data
public class CustUser implements Cloneable{
private String firstName;
private String lastName;
private Address address;
private String[] cars; @Override
public Object clone() throws CloneNotSupportedException{
return super.clone();
}
}

上面的例子中,我们定义了CustUser和Address。

 public void testShallowCopy() throws CloneNotSupportedException {
Address address= new Address();
address.setName("北京天安门");
CustUser custUser = new CustUser();
custUser.setAddress(address);
custUser.setLastName("李");
custUser.setFirstName("雷");
String[] cars = new String[]{"别克","路虎"};
custUser.setCars(cars); CustUser custUserCopy=(CustUser) custUser.clone();
custUserCopy.setFirstName("梅梅");
custUserCopy.setLastName("韩");
custUserCopy.getAddress().setName("北京颐和园");
custUserCopy.getCars()[0]="奥迪"; log.info("{}",custUser);
log.info("{}",custUserCopy);
}

浅拷贝我们只调用了CustUser的clone方法。看下输出结果:

CustUser(firstName=雷, lastName=李, address=Address(name=北京颐和园), cars=[奥迪, 路虎])

CustUser(firstName=梅梅, lastName=韩, address=Address(name=北京颐和园), cars=[奥迪, 路虎])

我们可以看到拷贝之后的Address变化会影响到被拷贝的对象。

上面的例子我们还要关注两个点:第一点String是不可变的。不管是拷贝还是赋值,String都是不可变的。

第二点,上面的例子中我们定义了一个数组,可以看到如果只是调用clone的话,数组也是浅拷贝。

使用clone的深拷贝

要使用深拷贝,只需要修改CustUser的构造函数就可以了:

//不是很好的使用方式
@Override
public Object clone() throws CloneNotSupportedException{
CustUserDeep custUserDeep=(CustUserDeep)super.clone();
custUserDeep.address=(Address)address.clone();
custUserDeep.cars=cars.clone();
return custUserDeep;
}

在重写的clone方法中,我们分别调用了CustUser,Address和数组的clone方法来进行拷贝。

再运行一次上面的测试代码:

CustUserDeep(firstName=雷, lastName=李, address=Address(name=北京天安门), cars=[别克, 路虎])

CustUserDeep(firstName=梅梅, lastName=韩, address=Address(name=北京颐和园), cars=[奥迪, 路虎])

可以看到address和cars是不同的,这表示我们的深拷贝是成功的。

不要overridden clone

上面的例子我们是通过overridden Object的clone方法来实现的。

但是最佳实践是不要overridden clone。那我们怎么做呢?

使用构造函数来构建新的对象:

    //好的方式
Address(Address address){
this.name=address.name;
}
//很好的方式
CustUserDeep(CustUserDeep custUserDeep){
this.firstName=custUserDeep.firstName;
this.lastName=custUserDeep.lastName;
this.cars=custUserDeep.getCars().clone();
this.address=new Address(custUserDeep.getAddress());
}

据说数组直接用clone来拷贝会更快,也可以使用下面的方式来拷贝数组:

this.cars= Arrays.copyOf(custUserDeep.getCars(),custUserDeep.getCars().length);

总结

本文讲解了浅拷贝和深拷贝的应用,并对clone方法做了深入的探讨。

本文的例子https://github.com/ddean2009/learn-java-base-9-to-20

本文作者:flydean程序那些事

本文链接:http://www.flydean.com/java-base-shallow-copy-deep-copy/

本文来源:flydean的博客

欢迎关注我的公众号:程序那些事,更多精彩等着您!

java深入理解浅拷贝和深拷贝的更多相关文章

  1. 【转】JAVA中的浅拷贝和深拷贝

    原文网址:http://blog.bd17kaka.net/blog/2013/06/25/java-deep-copy/ JAVA中的浅拷贝和深拷贝(shallow copy and deep co ...

  2. Java中的clone方法-理解浅拷贝和深拷贝

    最近学到Java虚拟机的相关知识,更加能理解clone方法的机制了 java中的我们常常需要复制的类型有三种: 1:8种基本类型,如int,long,float等: 2:复合数据类型(数组): 3:对 ...

  3. java中的浅拷贝和深拷贝

    复制 将一个对象的引用复制给另一个对象,一共有三种方式.第一种方式是直接赋值,第二种方式是浅复制,第三种方式是深复制. 1.直接赋值 在Java中,A a1 = a2,这实际上复制的是引用,也就是说 ...

  4. java引用类型的浅拷贝与深拷贝理解

    1.浅拷贝 只会复制地址值,也就是同一个对象两个引用,只是复制了一个引用而已. 2.深拷贝 重新在堆里创建一个新对象给新引用,连同地址值也不一样. 首先要知道Object的clone()方法, pub ...

  5. java中的浅拷贝与深拷贝

    浅拷贝: package test; class Student implements Cloneable { private int number; public int getNumber() { ...

  6. Java对象的浅拷贝和深拷贝&&String类型的赋值

    Java中的数据类型分为基本数据类型和引用数据类型.对于这两种数据类型,在进行赋值操作.方法传参或返回值时,会有值传递和引用(地址)传递的差别. 浅拷贝(Shallow Copy): ①对于数据类型是 ...

  7. Java 数组的浅拷贝和深拷贝

    浅拷贝: 在堆内存中不会分配新的空间,而是增加一个引用变量和之前的引用指向相同的堆空间. int[] a = {1,2,3,4,5}; int[]b = a; public class Test { ...

  8. 【Java】 Java中的浅拷贝和深拷贝

    先抛出结论: 浅拷贝是引用拷贝,A对象拷贝B以后,A对象和B对象指向同一块内存地址,改变A对象的属性值会触发B对象属性的改变,有安全风险 深拷贝是对象拷贝,A对象拷贝B以后,A对象和B对象指向不同的额 ...

  9. Python中变量、赋值、浅拷贝、深拷贝

    https://www.cnblogs.com/LetMe/p/6724555.html 在理解浅拷贝和深拷贝之前,首先要理解学习一下变量在Python中是怎样存储的: 变量的类型是分值引用与地址引用 ...

  10. python赋值、浅拷贝、深拷贝区别

    在写Python过程中,经常会遇到对象的拷贝,如果不理解浅拷贝和深拷贝的概念,你的代码就可能出现一些问题.所以,在这里按个人的理解谈谈它们之间的区别. 一.赋值(assignment) 在<Py ...

随机推荐

  1. 【Azure Redis 缓存 Azure Cache For Redis】Redis支持的版本及不同版本迁移风险

    问题描述 1. Azure Redis缓存支持的版本包括4.0以及6.0(预览) 这种情形下,可以使用PaaS服务提供的 Azure Redis 缓存(4.0版本).Azure Redis对6.0的支 ...

  2. 【Azure 应用服务】Azure App Service (Windows) 使用Flask框架部署Python应用,如何在代码中访问静态文件呢?如何设置文件路径?是相对路径还是绝对路径呢?

    问题描述 使用Flask框架部署Python代码,如何访问其中的静态文件呢?如static问价夹中的图像资源,同时如何在代码中读取txt文件中的内容呢?是相对路径或者是绝对路径呢? 实验步骤 在App ...

  3. Java UML类图

    在UML的静态机制中类图是一个重点,它不但是设计人员关心的核心,更是实现人员关注的核心.建模工具也主要根据类图来产生代码.类图在UML的9个图中占据了一个相当重要的地位.James Rumbaugh对 ...

  4. Redis哨兵模式搭建

    一:哨兵主要作用 监控:监控redis主库及从库运行状态: 通知:如果redis发生故障转移,可以通过邮件通知管理员: 自动故障转移:一旦发现主库宕机,则在从库中通过选举新的master进行故障转移. ...

  5. Codeforces Round 829 (Div. 1)A1. Make Nonzero Sum (easy version)(思维找规律)

    先考虑无解的情况:当n为奇数时无解 相邻的两个元素一定可以变成0 \[a[i] != a[i + 1]时, 分成[i, i], 和[i + 1, i + 1] \] \[a[i] = a[i + 1] ...

  6. Redis稳定性之战:AOF日志支撑数据持久化

    ★ Redis24篇集合 1 介绍 AOF(Append Only File)持久化:以独立日志的方式存储了 Redis 服务器的顺序指令序列,并只记录对内存进行修改的指令. 当Redis服务发生雪崩 ...

  7. autohotkey 设置快捷键 设置光标位置 (ctrl + alt + Numpad0)

    autohotkey 设置快捷键 设置光标位置 (ctrl + alt + Numpad0) 原因 3个屏幕,所以鼠标设置的灵敏度非常高,经常就找不到鼠标在哪了. 设置个快捷键,让鼠标每次都初始化一个 ...

  8. tomcat 安装笔记 20230901

    war位置 /usr/local/tomcat8_1/webapps/ tomcat位置 71.170 /usr/local/tomcat8_1/ 给了点工具包 位置 /usr/local/tool/ ...

  9. idea dev 分支合并到 master 流程

    合并分支前要全部提交 包括config.js 要不merge时候很麻烦 1 切换到master分支 Checkout 2 Merge into Current 3 commit push 4 切换回d ...

  10. The History of the English language 英语语音的起源 - 英语的历史 - 古英语 印欧语 希腊语 拉丁语

    印欧语 希腊语 拉丁语 日耳曼语 都是什么年代的语言 https://time.graphics/line/776755 印欧语是指印欧语系,它最初被认为在公元前2500年左右分散流传于欧洲.亚洲和印 ...