探讨Java中的深浅拷贝问题


Posted in Java/Android onJune 26, 2021

一、前言

拷贝这个词想必大家都很熟悉,在工作中经常需要拷贝一份文件作为副本。拷贝的好处也很明显,相较于新建来说,可以节省很大的工作量。在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='市场部'}}

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

以上就是探讨Java中的深浅拷贝问题的详细内容,更多关于Java深浅拷贝的资料请关注三水点靠木其它相关文章!

Java/Android 相关文章推荐
手把手教你用SpringBoot将文件打包成zip存放或导出
Jun 11 Java/Android
详解Spring Boot使用系统参数表提升系统的灵活性
Jun 30 Java/Android
Java生成读取条形码和二维码的简单示例
Jul 09 Java/Android
dubbo集成zipkin获取Traceid的实现
Jul 26 Java/Android
简述Java中throw-throws异常抛出
Aug 07 Java/Android
java如何实现socket连接方法封装
Sep 25 Java/Android
maven依赖的version声明控制方式
Jan 18 Java/Android
Netty客户端接入流程NioSocketChannel创建解析
Mar 25 Java/Android
详解Flutter网络请求Dio库的使用及封装
Apr 14 Java/Android
详解Android中的TimePickerView(时间选择器)的用法
Apr 30 Java/Android
Java Spring Lifecycle的使用
May 06 Java/Android
Android RecyclerView实现九宫格效果
Jun 28 Java/Android
解决SpringBoot跨域的三种方式
Jun 26 #Java/Android
分析Java中Map的遍历性能问题
Jun 26 #Java/Android
SpringCloud的JPA连接PostgreSql的教程
spring项目中切面及AOP的使用方法
Java 中的 Unsafe 魔法类的作用大全
Jun 26 #Java/Android
详解Java线程池是如何重复利用空闲线程的
Jun 26 #Java/Android
Spring Data JPA的Audit功能审计数据库的变更
You might like
PHP 开发工具
2006/12/06 PHP
php实现的太平洋时间和北京时间互转的自定义函数分享
2014/08/19 PHP
php中的mongodb select常用操作代码示例
2014/09/06 PHP
PHP实现用户异地登录提醒功能的方法【基于thinkPHP框架】
2018/03/15 PHP
wordpress自定义标签云与随机获取标签的方法详解
2019/03/22 PHP
jQuery textarea的长度进行验证
2009/05/06 Javascript
jquery 学习之二 属性(类)
2010/11/25 Javascript
Javascript判断对象是否相等实现代码
2013/03/18 Javascript
ExtJs默认的字体大小改变的几种方法(自己整理)
2013/04/18 Javascript
基于JavaScript自定义构造函数的详解说明
2013/04/24 Javascript
页面元素绑定jquery toggle后元素隐藏的解决方法
2014/03/27 Javascript
基于jquery实现百度新闻导航菜单滑动动画
2016/03/15 Javascript
对jquery的ajax进行二次封装以及ajax缓存代理组件:AjaxCache详解
2016/04/11 Javascript
node.js cookie-parser 中间件介绍
2016/06/06 Javascript
JavaScript实现窗口抖动效果
2016/10/19 Javascript
JS实现的数字格式化功能示例
2017/02/10 Javascript
BootStrap数据表格实例代码
2017/09/13 Javascript
node版本管理工具n包使用教程详解
2018/11/09 Javascript
解决node.js含有%百分号时发送get请求时浏览器地址自动编码的问题
2019/11/20 Javascript
微信小程序实现时间戳格式转换
2020/07/20 Javascript
Python httplib,smtplib使用方法
2008/09/06 Python
Python语言的12个基础知识点小结
2014/07/10 Python
Python和JavaScript间代码转换的4个工具
2016/02/22 Python
用Pytorch训练CNN(数据集MNIST,使用GPU的方法)
2019/08/19 Python
基于python的docx模块处理word和WPS的docx格式文件方式
2020/02/13 Python
QML实现钟表效果
2020/06/02 Python
请编写一个 C 函数,该函数在给定的内存区域搜索给定的字符,并返回该字符所在位置索引值
2014/09/15 面试题
法院实习人员自我鉴定
2013/09/26 职场文书
外国人聘用意向书
2014/04/01 职场文书
小学教师岗位职责
2015/04/02 职场文书
工程催款通知书
2015/04/17 职场文书
淮海战役观后感
2015/06/11 职场文书
团结友爱主题班会
2015/08/13 职场文书
大学学习委员竞选稿
2015/11/20 职场文书
MySQL 重写查询语句的三种策略
2021/05/10 MySQL
MYSQL如何查看进程和kill进程
2022/03/13 MySQL