探讨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 相关文章推荐
Feign调用全局异常处理解决方案
Jun 24 Java/Android
死磕 java同步系列之synchronized解析
Jun 28 Java/Android
SpringBoot 拦截器妙用你真的了解吗
Jul 01 Java/Android
零基础学java之循环语句的使用
Apr 10 Java/Android
Java8利用Stream对列表进行去除重复的方法详解
Apr 14 Java/Android
java版 联机五子棋游戏
May 04 Java/Android
Java 轮询锁使用时遇到问题
May 11 Java/Android
Android开发手册Chip监听及ChipGroup监听
Jun 10 Java/Android
Java 多线程并发FutureTask
Jun 28 Java/Android
SpringBoot项目部署到阿里云服务器的实现步骤
Jun 28 Java/Android
SpringBoot详解整合Redis缓存方法
Jul 15 Java/Android
前端与RabbitMQ实时消息推送未读消息小红点实现示例
Jul 23 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中的MVC模式运用技巧
2007/05/03 PHP
php at(@)符号的用法简介
2009/07/11 PHP
php 提速工具eAccelerator 配置参数详解
2010/05/16 PHP
PHP CURL采集百度搜寻结果图片不显示问题的解决方法
2017/02/03 PHP
yii插入数据库防并发的简单代码
2017/05/27 PHP
PHP实现的折半查找算法示例
2017/12/19 PHP
PHP开发之用微信远程遥控服务器
2018/01/25 PHP
Laravel Eloquent ORM 实现查询表中指定的字段
2019/10/17 PHP
PhpStorm连接服务器并实现自动上传功能
2020/12/09 PHP
DOM Scripting中的图片切换[兼容Firefox]
2010/06/12 Javascript
juqery 学习之三 选择器 子元素与表单
2010/11/25 Javascript
javascript移动设备Web开发中对touch事件的封装实例
2014/06/05 Javascript
Jquery中基本选择器用法实例详解
2015/05/18 Javascript
JavaScript操作class和style样式代码详解
2016/02/13 Javascript
JavaScript必知必会(三) String .的方法来自何方
2016/06/08 Javascript
jQuery实现表格行和列的动态添加与删除方法【测试可用】
2016/08/01 Javascript
AngularJS基础 ng-if 指令用法
2016/08/01 Javascript
BOM系列第一篇之定时器setTimeout和setInterval
2016/08/17 Javascript
jquery实现左右滑动式轮播图
2017/03/02 Javascript
浅谈Vue父子组件和非父子组件传值问题
2017/08/22 Javascript
React Router V4使用指南(精讲)
2018/09/17 Javascript
详解Vue Elementui中的Tag与页面其它元素相互交互的两三事
2018/09/25 Javascript
基于Vue+elementUI实现动态表单的校验功能(根据条件动态切换校验格式)
2019/04/04 Javascript
WebGL学习教程之Three.js学习笔记(第一篇)
2019/04/25 Javascript
微信小程序8种数据通信的方式小结
2020/02/03 Javascript
[02:12]2019完美世界全国高校联赛(春季赛)报名开启
2019/03/01 DOTA
浅谈Django学习migrate和makemigrations的差别
2018/01/18 Python
使用OpCode绕过Python沙箱的方法详解
2019/09/03 Python
详解python statistics模块及函数用法
2019/10/27 Python
python+Django+pycharm+mysql 搭建首个web项目详解
2019/11/29 Python
keras模型可视化,层可视化及kernel可视化实例
2020/01/24 Python
Mansur Gavriel官网:纽约市的一个设计品牌
2019/05/02 全球购物
关于雷锋的演讲稿
2014/05/10 职场文书
校友回访母校寄语
2015/02/26 职场文书
试了下Golang实现try catch的方法
2021/07/01 Golang
python非标准时间的转换
2021/07/25 Python