探讨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 相关文章推荐
Java数据结构之链表相关知识总结
Jun 18 Java/Android
Netty结合Protobuf进行编解码的方法
Jun 26 Java/Android
Java基础之this关键字的使用
Jun 30 Java/Android
分析ZooKeeper分布式锁的实现
Jun 30 Java/Android
SpringBoot工程下使用OpenFeign的坑及解决
Jul 02 Java/Android
Java比较两个对象中全部属性值是否相等的方法
Aug 07 Java/Android
关于@OnetoMany关系映射的排序问题,使用注解@OrderBy
Dec 06 Java/Android
Android超详细讲解组件ScrollView的使用
Mar 31 Java/Android
Spring Boot 底层原理基础深度解析
Apr 03 Java/Android
Android Flutter实现图片滑动切换效果
Apr 07 Java/Android
JAVA长虹键法之建造者Builder模式实现
Apr 10 Java/Android
Spring JPA 增加字段执行异常问题及解决
Jun 10 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微信开发之查询城市天气
2016/06/23 PHP
php自定义函数实现二维数组按指定key排序的方法
2016/09/29 PHP
PHP操作Redis常用命令的实例详解
2020/12/23 PHP
基于jquery的direction图片渐变动画效果
2010/05/24 Javascript
jquery多浏览器捕捉回车事件代码
2010/06/22 Javascript
jQuery中index()的用法分析
2014/09/05 Javascript
简单谈谈javascript中this的隐式绑定
2016/02/22 Javascript
jQuery.ajax实现根据不同的Content-Type做出不同的响应
2016/11/03 Javascript
react-native中ListView组件点击跳转的方法示例
2017/09/30 Javascript
jquery应用实例分享_实现手风琴特效
2018/02/01 jQuery
详解node.js 下载图片的 2 种方式
2018/03/02 Javascript
vue watch普通监听和深度监听实例详解(数组和对象)
2018/08/16 Javascript
详解vue文件中使用echarts.js的两种方式
2018/10/18 Javascript
JS选取DOM元素常见操作方法实例分析
2018/12/10 Javascript
解决layui页面按钮点击无反应,也不报错的问题
2019/09/29 Javascript
vue实现列表滚动的过渡动画
2020/06/29 Javascript
element-ui和vue表单(对话框)验证提示语(残留)清除操作
2020/09/11 Javascript
[01:35]2018完美盛典章节片——共竞
2018/12/17 DOTA
爬山算法简介和Python实现实例
2014/04/26 Python
python执行等待程序直到第二天零点的方法
2015/04/23 Python
Python模块包中__init__.py文件功能分析
2016/06/14 Python
pandas的object对象转时间对象的方法
2018/04/11 Python
浅谈django三种缓存模式的使用及注意点
2018/09/30 Python
python的schedule定时任务模块二次封装方法
2019/02/19 Python
TensorFlow加载模型时出错的解决方式
2020/02/06 Python
浅谈pytorch中的BN层的注意事项
2020/06/23 Python
一款利用html5和css3实现的3D滚动特效的教程
2015/01/04 HTML / CSS
英国顶级家庭折扣店:The Works
2017/09/06 全球购物
C++面试题:关于链表和指针
2013/06/05 面试题
Overload和Override的区别
2012/09/02 面试题
结婚喜宴家长答谢词
2014/01/15 职场文书
二年级数学教学反思
2014/01/21 职场文书
市场营销工作计划书
2014/09/15 职场文书
论群众路线学习笔记
2014/11/06 职场文书
环保建议书作文500字
2015/09/14 职场文书
2016年教师政治思想表现评语
2015/12/02 职场文书