探讨Java中的深浅拷贝问题

作者:Java旅途 时间:2023-01-07 01:12:44 

一、前言

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

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


protected native Object clone() throws CloneNotSupportedException;

1.该方法由 protected 修饰,java中所有类默认是继承Object类的,重载后的clone()方法为了保证其他类都可以正常调用,修饰符需要改成public

2.该方法是一个native方法,被native修饰的方法实际上是由非Java代码实现的,效率要高于普通的java方法。

3.该方法的返回值是Object对象,因此我们需要强转成我们需要的类型。

4.该方法抛出了一个CloneNotSupportedException异常,意思就是不支持拷贝,需要我们实现Cloneable接口来标记,这个类支持拷贝。

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

Dept类:


@Data
@AllArgsConstructor
@NoArgsConstructor
public class Dept {

private int deptNo;
   private String name;
}

User类:


@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {

private int age;
   private String name;
   private Dept dept;
}

二、浅拷贝

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

探讨Java中的深浅拷贝问题

浅拷贝功能实现


@Data
@AllArgsConstructor
@NoArgsConstructor
public class User implements Cloneable{

private int age;
   private String name;
   private Dept dept;
   @Override
   protected Object clone() throws CloneNotSupportedException {
       return super.clone();
   }
}

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


public static void main(String[] args) throws Exception{

Dept dept = new Dept(12, "市场部");
   User user = new User(18, "Java旅途", dept);

User user1 = (User)user.clone();
   System.out.println(user == user1);
   System.out.println();

user1.setAge(20);
   System.out.println(user);
   System.out.println(user1);
   System.out.println();

dept.setName("研发部");
   System.out.println(user);
   System.out.println(user1);
}

上面代码的运行结果如下

false

User{age=18, name='Java', dept=Dept{deptNo=12, name='市场部'}}

User{age=20, name='Java', dept=Dept{deptNo=12, name='市场部'}}

User{age=18, name='Java', dept=Dept{deptNo=12, name='研发部'}}

User{age=20, name='Java', dept=Dept{deptNo=12, name='研发部'}}

三、深拷贝

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

探讨Java中的深浅拷贝问题

深拷贝功能实现

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

dept类:


@Data
@AllArgsConstructor
@NoArgsConstructor
public class Dept implements Cloneable {

private int deptNo;
   private String name;

@Override
   public Object clone() throws CloneNotSupportedException {
       return super.clone();
   }
}

user类:


@Data
@AllArgsConstructor
@NoArgsConstructor
public class User implements Cloneable{

private int age;
   private String name;
   private Dept dept;

@Override
   protected Object clone() throws CloneNotSupportedException {
       User user = (User) super.clone();
       user.dept =(Dept) dept.clone();
       return user;
   }
}

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

false

User{age=18, name='Java旅途', dept=Dept{deptNo=12, name='市场部'}}

User{age=20, name='Java旅途', dept=Dept{deptNo=12, name='市场部'}}

User{age=18, name='Java旅途', dept=Dept{deptNo=12, name='研发部'}}

User{age=20, name='Java旅途', dept=Dept{deptNo=12, name='市场部'}}

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

来源:https://www.cnblogs.com/zhixie/p/14323689.html

标签:Java,深拷贝,浅拷贝
0
投稿

猜你喜欢

  • C#简单了解接口(Interface)使用方法

    2022-01-30 18:33:57
  • java实现摄像头截图功能

    2023-12-01 19:53:22
  • C#实现左截取和右截取字符串实例

    2022-06-06 08:29:39
  • SpringBoot在生产快速禁用Swagger2的方法步骤

    2022-12-30 00:05:53
  • Spring加载properties文件的两种方式实例详解

    2021-11-20 23:36:28
  • Java如何实现单链表的增删改查

    2021-09-19 09:49:25
  • C#实现winform自动关闭MessageBox对话框的方法

    2022-09-02 02:21:53
  • 自定义Spring Security的身份验证失败处理方法

    2022-09-27 13:18:31
  • Spring Boot 2.0 配置属性自定义转换的方法

    2021-10-18 12:07:44
  • Java与Oracle实现事务(JDBC事务)实例详解

    2023-02-12 12:47:22
  • Android标题栏中添加返回按钮功能

    2022-09-16 05:52:47
  • 深入多线程之:双向信号与竞赛的用法分析

    2022-02-17 06:54:49
  • Android实现Banner界面广告图片循环轮播(包括实现手动滑动循环)

    2022-02-06 17:05:35
  • 详解Spring Boot微服务如何集成fescar解决分布式事务问题

    2022-01-21 18:43:51
  • JDK源码分析之String、StringBuilder和StringBuffer

    2022-01-23 00:10:20
  • Android 基于IntentService的文件下载的示例代码

    2023-10-28 23:26:02
  • Java守护线程实例详解_动力节点Java学院整理

    2023-03-29 08:14:30
  • 浅谈图片上传利用request.getInputStream()获取文件流时遇到的问题

    2023-10-18 10:36:43
  • SpringBoot整合MongoDB的步骤详解

    2023-10-11 17:20:32
  • C# memcache 使用介绍

    2022-11-25 12:25:57
  • asp之家 软件编程 m.aspxhome.com