java序列化深克隆

在我以前的文章中,我解释了深度克隆和浅层克隆之间的区别 , 以及复制构造函数和防御性复制方法比默认的Java克隆如何更好。

使用复制构造函数和防御性复制方法进行的Java对象克隆当然具有某些优势,但是我们必须显式编写一些代码才能在所有这些方法中实现深度克隆。 而且,仍然有可能我们会错过某些东西并且不会得到深克隆的对象。

正如在Java中创建对象的5种不同方式中所讨论的那样,反序列化序列化对象会创建一个状态与序列化对象相同的新对象。 因此,与上述克隆方法类似,我们也可以使用对象序列化和反序列化来实现深度克隆功能,并且通过这种方法,我们不必担心或编写用于深度克隆的代码,默认情况下会得到它。

但是,使用序列化克隆对象会带来一些性能开销,如果我们只需要克隆对象并且不需要将其持久保存在文件中以备将来使用,则可以通过使用内存中序列化来改进它。

我们将使用以下Employee类作为示例,其name
作为状态的dojskills ,对于深度克隆,我们无需担心code> name字段,因为它是一个String对象,默认情况下所有
弦在本质上是不变的 。

您可以在《 如何在Java中创建不可变的类》以及《 为什么String是不可变的和Final》上阅读有关不可变性的更多信息。

 Employee class implements Serializable { private static final long serialVersionUID = 2L; private String name; private LocalDate doj; private List<String> skills; public Employee(String name, LocalDate doj, List<String> skills) { this .name = name; this .doj = doj; this .skills = skills; } public String getName() { return name; } name; } public LocalDate getDoj() { return doj; } doj; } public List<String> getSkills() { return skills; } skills; } // Method to deep clone a object using in memory serialization public Employee deepClone() throws IOException, ClassNotFoundException { // First serializing the object and its state to memory using ByteArrayOutputStream instead of FileOutputStream. ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream out = new ObjectOutputStream(bos); out.writeObject( this ); // And then deserializing it from memory using ByteArrayOutputStream instead of FileInputStream. // Deserialization process will create a new object with the same state as in the serialized object, ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); ObjectInputStream in = new ObjectInputStream(bis); return (Employee) in.readObject(); } @Override public String toString() { return String.format( "Employee{name='%s', doj=%s, skills=%s}" , name, doj, skills); } @Override public boolean equals(Object o) { if ( this == o) return true ; if (o == null || getClass() != o.getClass()) return false ; Employee employee = (Employee) o; return Objects.equals(name, employee.name) && Objects.equals(doj, employee.doj) && Objects.equals(skills, employee.skills); } @Override public int hashCode() { return Objects.hash(name, doj, skills); }  } 

为了深度克隆Employee类的对象,我提供了一个
deepClone()方法,通过使用将对象序列化到内存
ByteArrayOutputStream而不是FileOutputStream并使用ByteArrayInputStream而不是FileInputStream将其反序列化。 在这里,我们将对象序列化为字节,然后再次将其从字节反序列化为对象。


Employee类实现Serializable接口来实现序列化,这有其自身的劣势,我们可以通过使用Externalizable接口来自定义序列化过程来克服其中的某些劣势。

我们可以在下面的测试中运行,以了解我们的克隆方法是深层克隆还是浅层克隆,此处所有==操作将返回false(因为两个对象是分开的),而所有equals将返回true(因为两者具有相同的内容)。

 public static void main(String[] args) throws IOException, ClassNotFoundException { Employee emp = new Employee( "Naresh Joshi" , LocalDate.now(), Arrays.asList( "Java" , "Scala" , "Spring" )); System.out.println( "Employee object: " + emp); // Deep cloning `emp` object by using our `deepClone` method. Employee clonedEmp = emp.deepClone(); System.out.println( "Cloned employee object: " + clonedEmp); System.out.println(); // All of this will print false because both objects are separate. System.out.println(emp == clonedEmp); System.out.println(emp.getDoj() == clonedEmp.getDoj()); System.out.println(emp.getSkills() == clonedEmp.getSkills()); System.out.println(); // All of this will print true because `clonedEmp` is a deep clone of `emp` and both have the same content. System.out.println(Objects.equals(emp, clonedEmp)); System.out.println(Objects.equals(emp.getDoj(), clonedEmp.getDoj())); System.out.println(Objects.equals(emp.getSkills(), clonedEmp.getSkills()));  } 

我们知道反序列化过程每次都会创建一个新对象,如果我们必须使我们的类单身,那将是不好的。 这就是为什么我们需要覆盖和禁用单例类的序列化,这可以通过提供writeReplace和readResolve方法来实现。

与序列化类似,Java克隆也不能与单例模式一起使用,这就是为什么我们也需要覆盖和禁用它。 我们可以通过实现克隆的方式来做到这一点,以便它要么抛出
CloneNotSupportedException或每次都返回相同的实例。

您可以在Java Cloning和Java上阅读有关Java克隆和序列化的更多信息。
Java序列化主题。

您可以在此找到本文的完整源代码。
Github存储库 ,请随时提供宝贵的反馈。

翻译自: https://www.javacodegeeks.com/2019/08/deep-clone-using-java-memory-serialization.html

java序列化深克隆

java序列化深克隆_如何在内存序列化中使用Java深克隆对象相关推荐

  1. java transient关键字_嗯?你真的会用 Java 序列化机制?

    码农每日一题长按关注,工作日每天分享一个技术知识点. 一.序列化.反序列化概念及其使用场景 1.序列化.反序列化的概念 简单的讲,序列化就是将java对象转化成二进制保存到磁盘中去,反序列化就是从磁盘 ...

  2. mysql 序列化存储_如何将类序列化并直接存储入数据库

    本文将从这两个格式器入手,先向大家介绍分别用它们如何实现序列化和反序列化,然后比较两种格式器的不同点.接着我会向大家介绍实现序列化对对象类型的一些要求,同时还要向大家介绍两种不同的序列化方式:基本序列 ...

  3. java eden区_(转)可能是把Java内存区域讲的最清楚的一篇文章

    写在前面 本节常见面试题: 问题答案在文中都有提到 如何判断对象是否死亡(两种方法). 简单的介绍一下强引用.软引用.弱引用.虚引用(虚引用与软引用和弱引用的区别.使用软引用能带来的好处). 如何判断 ...

  4. java gui框架_推荐!程序员整理的Java资源大全

    构建 这里搜集了用来构建应用程序的工具. Apache Maven:Maven使用声明进行构建并进行依赖管理,偏向于使用约定而不是配置进行构建.Maven优于Apache Ant.后者采用了一种过程化 ...

  5. 牛客网 java刷题_牛客网刷题(纯java题型 1~30题)

    应该是先extend,然后implement class test extends A implements B { public static void main(String[] args) { ...

  6. java hashmap用法_备战金九银十:Java核心技术面试题100+,助你搞定面试官

    一线互联网公司工作了几年,我作为求职者参加了不少面试,也作为面试官面试了很多同学,整理这份面试指南,一方面是帮助大家更好的准备面试,有的放矢,另一方面也是对自己知识框架做一个体系化的梳理. 这篇文章梳 ...

  7. java框架核心技术_你必须掌握的 21 个 Java 核心技术!(干货)

    点击上方"java进阶架构师",选择右上角"置顶公众号" 20大进阶架构专题每日送达 51闲来无事,师长一向不(没)喜(有)欢(钱)凑热闹,倒不如趁着这时候复盘 ...

  8. java oom分析_作为测试你应该知道的JAVA OOM及定位分析

    上周现网一个内存溢出问题导致应用服务器每隔一小时死一次,遂整理下常见的OMM.发现方法和处理方式,加入Bug预防. 常见的OutOfMemoryError有三种:OutOfMemoryError:Pe ...

  9. java future用法_你必须掌握的 21 个 Java 核心技术

    作者:工程师-搁浅来源:https://www.jb51.net/article/122070.htm 写这篇文章的目的是想总结一下自己这么多年来使用java的一些心得体会,主要是和一些java基础知 ...

最新文章

  1. windows 平台使用 VS2017 编译 libevent 源码
  2. php 处理raw数据,php以raw格式传递数据
  3. 短视频进入下半场,价值创造成赛点
  4. (十)java springboot b2b2c shop 多用户商城系统源码:服务网关zuul初级篇
  5. 投资学习网课笔记(part1)--基金第一课
  6. WP7应用开发笔记(3) 界面设计
  7. GNU make manual 翻译( 一百六十二)
  8. Mybatis笔记 – Po映射类型
  9. Atitit. visual studio vs2003 vs2005 vs2008  VS2010 vs2012 vs2015新特性 新功能.doc
  10. 《精通Javascript+jQuery》视频教程
  11. Java和Android笔试题
  12. android viewpager实现画廊效果,ViewPager实现2D、3D画廊效果
  13. 获取当前日期是今年的第几周
  14. 今天的几点感悟_20160711
  15. COMSOL如何绘制紧贴圆柱面的圆面
  16. linux 导出dump文件
  17. 代理服务器拒绝连接(无法连接到代理服务器)的解决办法
  18. 浅谈溯源反制与防溯源
  19. python爬取证券数据并存入数据库
  20. 独领风骚,卡兹特投影仪大热香港电子展

热门文章

  1. I - Triple HDU - 5517
  2. Acwing -- 单调队列优化的DP问题
  3. P4878 [USACO05DEC]Layout G
  4. 牛客题霸 转圈打印矩阵 C++题解/答案
  5. 牛客网 【每日一题】5月29日 管道取珠
  6. 中心城镇问题(长链剖分优化树形dp)
  7. 杯子 + Kronican
  8. AT3950-[AGC022E]Median Replace【贪心,dp】
  9. [2020.10.30NOIP模拟赛]字符串水题【SA,树状数组】
  10. nssl1317-灵魂分流药剂【分组背包,二维费用背包】