前言

Java 开发中,对象拷贝是常有的事,很多人可能搞不清到底是拷贝了引用还是拷贝了对象。本文将详细介绍相关知识,让你充分理解 Java 拷贝。


一、对象是如何存储的?

方法执行过程中,方法体中的数据类型主要分两种,它们的存储方式是不同的(如下图):

  1. 基本数据类型: 直接存储在栈帧的局部变量表中;
  2. 引用数据类型: 对象的引用存储在栈帧的局部变量表中,而对实例本身及其所有成员变量存放在堆内存中。

详情可见JVM基础

二、前置准备

创建两个实体类方便后续的代码示例

@Data
@AllArgsConstructor
public class Animal{
private int id;
private String type; @Override
public String toString () {
return "Animal{" +
"id=" + id +
", type='" + type + '\'' +
'}';
}
}
@Data
@AllArgsConstructor
public class Dog {
private int age;
private String name;
private Animal animal; @Override
public String toString () {
return "Dog{" +
"age=" + age +
", name='" + name + '\'' +
", animal=" + animal +
'}';
}
}

三、直接赋值

直接赋值是我们最常用的方式,它只是拷贝了对象引用地址,并没有在内存中生成新的对象

下面我们进行代码验证:

public class FuXing {
public static void main (String[] args) {
Animal animal = new Animal(1, "dog");
Dog dog = new Dog(18, "husky", animal);
Dog dog2 = dog;
System.out.println("两个对象是否相等:" + (dog2 == dog)); System.out.println("----------------------------");
dog.setAge(3);
System.out.println("变化后两个对象是否相等:" + (dog2 == dog));
}
}
两个对象是否相等:true
----------------------------
变化后两个对象是否相等:true

通过运行结果可知,dog类的age已经发生变化,但重新打印两个类依然相等。所以它只是拷贝了对象引用地址,并没有在内存中生成新的对象

直接赋值的 JVM 的内存结构大致如下:

四、浅拷贝

浅拷贝后会创建一个新的对象,且新对象的属性和原对象相同。但是,拷贝时针对原对象的属性的数据类型的不同,有两种不同的情况:

  1. 属性的数据类型基本类型,拷贝的就是基本类型的值;
  2. 属性的数据类型引用类型,拷贝的就是对象的引用地址,意思就是拷贝对象与原对象引用同一个对象

要实现对象浅拷贝还是比较简单的,只需要被拷贝的类实现Cloneable接口,重写clone方法即可。下面我们对Dog进行改动:

@Data
@AllArgsConstructor
public class Dog implements Cloneable{
private int age;
private String name;
private Animal animal; @Override
public Dog clone () throws CloneNotSupportedException {
return (Dog) super.clone();
} @Override
public String toString () {
return "Dog{" +
"age=" + age +
", name='" + name + '\'' +
", animal=" + animal +
'}';
}
}

接下来我们运行下面的代码,看一下运行结果:

public class FuXing {
public static void main (String[] args) throws Exception {
Animal animal = new Animal(1, "dog");
Dog dog = new Dog(18, "husky", animal); // 克隆对象
Dog cloneDog = dog.clone(); System.out.println("dog:" + dog);
System.out.println("cloneDog:" + cloneDog);
System.out.println("两个对象是否相等:" + (cloneDog == dog));
System.out.println("两个name是否相等:" + (cloneDog.getName() == dog.getName()));
System.out.println("两个animal是否相等:" + (cloneDog.getAnimal() == dog.getAnimal())); System.out.println("----------------------------------------"); // 更改原对象的属性值
dog.setAge(3);
dog.setName("corgi");
dog.getAnimal().setId(2); System.out.println("dog:" + dog);
System.out.println("cloneDog:" + cloneDog);
System.out.println("两个对象是否相等:" + (cloneDog == dog));
System.out.println("两个name是否相等:" + (cloneDog.getName() == dog.getName()));
System.out.println("两个animal是否相等:" + (cloneDog.getAnimal() == dog.getAnimal()));
}
dog:Dog{age=18, name='husky', animal=Animal{id=1, type='dog'}}
cloneDog:Dog{age=18, name='husky', animal=Animal{id=1, type='dog'}}
两个对象是否相等:false
两个name是否相等:true
两个animal是否相等:true
----------------------------------------
dog:Dog{age=3, name='corgi', animal=Animal{id=2, type='dog'}}
cloneDog:Dog{age=18, name='husky', animal=Animal{id=2, type='dog'}}
两个对象是否相等:false
两个name是否相等:false
两个animal是否相等:true

我们分析下运行结果,重点看一下 “两个name是否相等”,改动后变成 false.

这是因为StringInteger等包装类都是不可变的对象,当需要修改不可变对象的值时,需要在内存中生成一个新的对象来存放新的值,然后将原来的引用指向新的地址

这里dog对象的name属性已经指向一个新的对象,而cloneDogname属性仍然指向原来的对象,所以就不同了。

然后我们看下两个对象的animal属性,原对象属性值变动后,拷贝对象也跟着变动,这就是因为拷贝对象与原对象引用同一个对象

浅拷贝的 JVM 的内存结构大致如下:

五、深拷贝

与浅拷贝不同之处,深拷贝在对引用数据类型进行拷贝的时候,创建了一个新的对象,并且拷贝其成员变量。也就是说,深拷贝出来的对象,与原对象没有任何关联,是一个新的对象。

实现深拷贝有两种方式

1. 让每个引用类型属性都重写clone()方法

注意: 这里如果引用类型的属性或者层数太多了,代码量会变很大,所以一般不建议使用

@Data
@AllArgsConstructor
public class Animal implements Cloneable{
private int id;
private String type; @Override
protected Animal clone () throws CloneNotSupportedException {
return (Animal) super.clone();
} @Override
public String toString () {
return "Animal{" +
"id=" + id +
", type='" + type + '\'' +
'}';
}
}
@Data
@AllArgsConstructor
public class Dog implements Cloneable{
private int age;
private String name;
private Animal animal; @Override
public Dog clone () throws CloneNotSupportedException {
Dog clone = (Dog) super.clone();
clone.animal = animal.clone();
return clone;
} @Override
public String toString () {
return "Dog{" +
"age=" + age +
", name='" + name + '\'' +
", animal=" + animal +
'}';
}
}

我们再次运行浅拷贝部分的main方法,结果如下。

dog:Dog{age=18, name='husky', animal=Animal{id=1, type='dog'}}
cloneDog:Dog{age=18, name='husky', animal=Animal{id=1, type='dog'}}
两个对象是否相等:false
两个name是否相等:true
两个animal是否相等:false # 变为false
----------------------------------------
dog:Dog{age=3, name='corgi', animal=Animal{id=2, type='dog'}}
cloneDog:Dog{age=18, name='husky', animal=Animal{id=1, type='dog'}}
两个对象是否相等:false
两个name是否相等:false
两个animal是否相等:false # 变为false

2.序列化

序列化是将对象写到流中便于传输,而反序列化则是把对象从流中读取出来。我们可以利用对象的序列化产生克隆对象,然后通过反序列化获取这个对象。

@Data
@AllArgsConstructor
public class Animal implements Serializable {
private int id;
private String type; @Override
public String toString () {
return "Animal{" +
"id=" + id +
", type='" + type + '\'' +
'}';
}
}
@Data
@AllArgsConstructor
public class Dog implements Serializable {
private int age;
private String name;
private Animal animal; @SneakyThrows
@Override
public Dog clone () {
// 序列化
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(this); //反序列化
ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
ObjectInputStream ois = new ObjectInputStream(bis);
return (Dog) ois.readObject();
} @Override
public String toString () {
return "Dog{" +
"age=" + age +
", name='" + name + '\'' +
", animal=" + animal +
'}';
}
}

我们再次运行浅拷贝部分的main方法,结果如下。

dog:Dog{age=18, name='husky', animal=Animal{id=1, type='dog'}}
cloneDog:Dog{age=18, name='husky', animal=Animal{id=1, type='dog'}}
两个对象是否相等:false
两个name是否相等:false # 变为false
两个animal是否相等:false # 变为false
----------------------------------------
dog:Dog{age=3, name='corgi', animal=Animal{id=2, type='dog'}}
cloneDog:Dog{age=18, name='husky', animal=Animal{id=1, type='dog'}}
两个对象是否相等:false
两个name是否相等:false
两个animal是否相等:false # 变为false

深拷贝的 JVM 的内存结构大致如下:

Java 中的深拷贝和浅拷贝你了解吗?的更多相关文章

  1. 浅谈Java中的深拷贝和浅拷贝(转载)

    浅谈Java中的深拷贝和浅拷贝(转载) 原文链接: http://blog.csdn.net/tounaobun/article/details/8491392 假如说你想复制一个简单变量.很简单: ...

  2. 浅谈Java中的深拷贝和浅拷贝

    转载: 浅谈Java中的深拷贝和浅拷贝 假如说你想复制一个简单变量.很简单: int apples = 5; int pears = apples; 不仅仅是int类型,其它七种原始数据类型(bool ...

  3. 内功心法 -- Java中的深拷贝和浅拷贝

    写在前面的话:读书破万卷,编码如有神--------------------------------------------------------------------这篇博客主要来谈谈" ...

  4. Java中的深拷贝和浅拷贝

    1.浅拷贝与深拷贝概念 (1)浅拷贝(浅克隆) 浅拷贝又叫浅复制,将对象中的所有字段复制到新的对象(副本)中.其中,值类型字段(java中8中原始类型)的值被复制到副本中后,在副本中的修改不会影响到源 ...

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

    Java中对象的创建 clone顾名思义就是复制, 在Java语言中, clone方法被对象调用,所以会复制对象.所谓的复制对象,首先要分配一个和源对象同样大小的空间,在这个空间中创建一个新的对象.那 ...

  6. Java中的深拷贝和浅拷贝(转载)

    深拷贝(深复制)和浅拷贝(浅复制)是两个比较通用的概念,尤其在C++语言中,若不弄懂,则会在delete的时候出问题,但是我们在这幸好用的是Java.虽然java自动管理对象的回收,但对于深拷贝(深复 ...

  7. java基础(十七)----- 浅谈Java中的深拷贝和浅拷贝 —— 面试必问

    假如说你想复制一个简单变量.很简单: int apples = 5; int pears = apples; 不仅仅是int类型,其它七种原始数据类型(boolean,char,byte,short, ...

  8. 浅析Java中的深拷贝和浅拷

      浅析Java中的深拷贝和浅拷贝 原文链接: http://blog.csdn.net/tounaobun/article/details/8491392 假如说你想复制一个简单变量.很简单: in ...

  9. **Python中的深拷贝和浅拷贝详解

    Python中的深拷贝和浅拷贝详解   这篇文章主要介绍了Python中的深拷贝和浅拷贝详解,本文讲解了变量-对象-引用.可变对象-不可变对象.拷贝等内容.   要说清楚Python中的深浅拷贝,需要 ...

  10. C语言中的深拷贝和浅拷贝

    //C语言中的深拷贝和浅拷贝 #define _CRT_SECURE_NO_WARNINGS #include<stdio.h> #include<stdlib.h> #inc ...

随机推荐

  1. 5. Determinant

    5.1 The Properties of Determinants The determinant of the n by n identity matrix is 1 : \(det I = 1\ ...

  2. Python调用动态库,获取BSTR字符串

    今天客户在用Python调用我们的动态库的时候,遇到一个问题,调用动态库中的函数,函数返回的是BSTR字符串,但是客户接收到的是一个8位长度的数字. 动态库函数原型:EXTERN_C BSTR ELO ...

  3. 基于istio实现多集群流量治理

    本文分享自华为云社区<基于istio实现多集群流量治理>,作者: 可以交个朋友. 一 背景 对多云.混合云等异构基础设施的服务治理是Istio重点支持的场景之一.为了提高服务的可用性,避免 ...

  4. Spring Boot 项目五维度九层次分层架构实现实践研究——持续更新中

    var code = "12433d02-b242-4fd2-937d-750761a365ea" 说明:本博文有参考一些技术博主的思路,据实践内容及代码持续总结更新中. 五个分层 ...

  5. web 报表工具如何自适应

    现在的报表用户已经不再将报表作为一个单纯的报表工具看待了,有时候也会当作页面工具来使用,这时为了页面显示工整美观,就需要报表能够自适应宽度.下面我们就基于润乾报表来讲一下是如何做到自适应展现报表. 产 ...

  6. docker 应用篇————具名挂载和匿名挂载[十三]

    前言 简单整理一下具名挂载和匿名挂载. 正文 来看一下匿名挂载. 这里-v指定了容器内部的路径,但是没有指定容器外部的路径,那么挂载到了什么地方. 用inspect 查看一下. 挂载到这个位置了. 然 ...

  7. MaxCompute Tunnel 技术原理及开发实战

    简介: MaxCompute(原名ODPS)是一种快速.完全托管的EB级数据仓库解决方案, 致力于批量结构化数据的存储和计算,为用户提供数据仓库的解决方案及分析建模服务.Tunnel是MaxCompu ...

  8. 无缝融入 Kubernetes 生态 | 云原生网关支持 Ingress 资源

    ​简介:Kubernetes 一贯的作风是通过定义标准来解决同一类问题,在解决集群对外流量管理的问题也不例外.Kubernetes 对集群入口点进行了进一步的统一抽象,提出了 3 种解决方案:Node ...

  9. Apache Flink在 bilibili 的多元化探索与实践

    简介: bilibili 万亿级传输分发架构的落地,以及 AI 领域如何基于 Flink 打造一套完善的预处理实时 Pipeline. 本文由 bilibili 大数据实时平台负责人郑志升分享,本次分 ...

  10. MaxCompute中如何通过logview诊断慢作业

    ​建模服务,在MaxCompute执行sql任务的时候有时候作业会很慢,本文通过查看logview排查具体任务慢的原因 在这里把任务跑的慢的问题划分为以下几类 资源不足导致的排队(一般是包年包月项目) ...