Java中的深浅拷贝问题,你清楚吗?

开发 后端
拷贝这个词想必大家都很熟悉,在工作中经常需要拷贝一份文件作为副本。拷贝的好处也很明显,相较于新建来说,可以节省很大的工作量。在Java中,同样存在拷贝这个概念,拷贝的意义也是可以节省创建对象的开销。

 [[374290]]

一、前言

拷贝这个词想必大家都很熟悉,在工作中经常需要拷贝一份文件作为副本。拷贝的好处也很明显,相较于新建来说,可以节省很大的工作量。在Java中,同样存在拷贝这个概念,拷贝的意义也是可以节省创建对象的开销。

Object类中有一个方法clone(),具体方法如下:

  1. protected native Object clone() throws CloneNotSupportedException; 
  1. 该方法由 protected 修饰,java中所有类默认是继承Object类的,重载后的clone()方法为了保证其他类都可以正常调用,修饰符需要改成public。
  2. 该方法是一个native方法,被native修饰的方法实际上是由非Java代码实现的,效率要高于普通的java方法。
  3. 该方法的返回值是Object对象,因此我们需要强转成我们需要的类型。
  4. 该方法抛出了一个CloneNotSupportedException异常,意思就是不支持拷贝,需要我们实现Cloneable接口来标记,这个类支持拷贝。

为了演示方便,我们新建两个实体类Dept 和 User,其中User依赖了Dept,实体类代码如下:

Dept类:

  1. @Data 
  2. @AllArgsConstructor 
  3. @NoArgsConstructor 
  4. public class Dept { 
  5.  
  6.     private int deptNo; 
  7.     private String name

User类:

  1. @Data 
  2. @AllArgsConstructor 
  3. @NoArgsConstructor 
  4. public class User { 
  5.  
  6.     private int age; 
  7.     private String name
  8.     private Dept dept; 

二、浅拷贝

对于基本类型的的属性,浅拷贝会将属性值复制给新的对象,而对于引用类型的属性,浅拷贝会将引用复制给新的对象。而像String,Integer这些引用类型,都是不可变的,拷贝的时候会创建一份新的内存空间来存放值,并且将新的引用指向新的内存空间。不可变类型是特殊的引用类型,我们姑且认为这些被final标记的引用类型也是复制值。

浅拷贝功能实现

  1. @Data 
  2. @AllArgsConstructor 
  3. @NoArgsConstructor 
  4. public class User implements Cloneable{ 
  5.  
  6.     private int age; 
  7.     private String name
  8.     private Dept dept; 
  9.     @Override 
  10.     protected Object clone() throws CloneNotSupportedException { 
  11.         return super.clone(); 
  12.     } 

如何验证我们的结论呢?首先对比被拷贝出的对象和原对象是否相等,不等则说明是新拷贝出的一个对象。其次修改拷贝出对象的基本类型属性,如果原对象的此属性发生了修改,则说明基本类型的属性是同一个,最后修改拷贝出对象的引用类型对象即Dept属性,如果原对象的此属性发生了改变,则说明引用类型的属性是同一个。清楚测试原理后,我们写一段测试代码来验证我们的结论。

  1. public static void main(String[] args) throws Exception{ 
  2.  
  3.     Dept dept = new Dept(12, "市场部"); 
  4.     User user = new User(18, "Java旅途", dept); 
  5.  
  6.     User user1 = (User)user.clone(); 
  7.     System.out.println(user == user1); 
  8.     System.out.println(); 
  9.  
  10.     user1.setAge(20); 
  11.     System.out.println(user); 
  12.     System.out.println(user1); 
  13.     System.out.println(); 
  14.  
  15.     dept.setName("研发部"); 
  16.     System.out.println(user); 
  17.     System.out.println(user1); 

上面代码的运行结果如下:

  1. false 
  2.  
  3. User{age=18, name='Java', dept=Dept{deptNo=12, name='市场部'}} 
  4. User{age=20, name='Java', dept=Dept{deptNo=12, name='市场部'}} 
  5.  
  6. User{age=18, name='Java', dept=Dept{deptNo=12, name='研发部'}} 
  7. User{age=20, name='Java', dept=Dept{deptNo=12, name='研发部'}} 

三、深拷贝

相较于浅拷贝而言,深拷贝除了会将基本类型的属性复制外,还会将引用类型的属性也会复制。

深拷贝功能实现

在拷贝user的时候,同时将user中的dept属性进行拷贝。

dept类:

  1. @Data 
  2. @AllArgsConstructor 
  3. @NoArgsConstructor 
  4. public class Dept implements Cloneable { 
  5.  
  6.     private int deptNo; 
  7.     private String name
  8.  
  9.     @Override 
  10.     public Object clone() throws CloneNotSupportedException { 
  11.         return super.clone(); 
  12.     } 

user类:

  1. @Data 
  2. @AllArgsConstructor 
  3. @NoArgsConstructor 
  4. public class User implements Cloneable{ 
  5.  
  6.     private int age; 
  7.     private String name
  8.     private Dept dept; 
  9.  
  10.     @Override 
  11.     protected Object clone() throws CloneNotSupportedException { 
  12.         User user = (User) super.clone(); 
  13.         user.dept =(Dept) dept.clone(); 
  14.         return user
  15.     } 

使用浅拷贝的测试代码继续测试,运行结果如下:

  1. false 
  2.  
  3. User{age=18, name='Java旅途', dept=Dept{deptNo=12, name='市场部'}} 
  4. User{age=20, name='Java旅途', dept=Dept{deptNo=12, name='市场部'}} 
  5.  
  6. User{age=18, name='Java旅途', dept=Dept{deptNo=12, name='研发部'}} 
  7. User{age=20, name='Java旅途', dept=Dept{deptNo=12, name='市场部'}} 

除此之外,还可以利用反序列化实现深拷贝,先将对象序列化成字节流,然后再将字节流序列化成对象,这样就会产生一个新的对象。

本文转载自微信公众号「 Java旅途」,可以通过以下二维码关注。转载本文请联系 Java旅途公众号。

 

责任编辑:武晓燕 来源: Java旅途
相关推荐

2020-12-01 11:33:57

Python拷贝copy

2023-05-12 08:11:58

JavaScriptJSON克隆

2011-06-03 17:14:35

iphone Objective

2022-09-26 09:01:23

JavaScript浅拷贝深拷贝

2023-11-08 14:21:51

Python拷贝

2020-12-18 06:09:07

Java浅拷贝深拷贝

2020-09-17 14:04:32

拷贝

2023-11-10 10:51:15

Python

2010-09-01 09:48:32

DHCP报文格式

2010-11-01 14:45:35

云计算

2010-08-20 09:46:52

云计算SaaS

2021-09-13 13:05:05

Redis数据库内存

2019-06-18 15:57:25

HTTP缓存机制

2009-05-19 17:28:44

深拷贝浅拷贝clone()

2019-09-23 08:46:04

零拷贝 CPU内存

2018-11-05 11:22:19

2020-08-06 11:05:30

函数调用寄存器语言

2023-08-04 08:25:03

客户配置Spring

2023-02-27 23:45:09

MySQL索引存储

2018-09-29 15:34:34

JavaList接口
点赞
收藏

51CTO技术栈公众号