java 序列化 父子类

2024-08-27 16:08
文章标签 java 序列化 父子

本文主要是介绍java 序列化 父子类,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

环境

java:1.7+

前言

今天读了这篇文章,一篇译文:
我下面的讲解就是基于这篇文章
Java序列化示例教程

到了晚上又看到了这么一篇文章:
Java 序列化的高级认识
中的对敏感字段加密

突然对其中某段代码产生了兴趣;现在记录下


回顾

我先记录下我今天看到的知识,看看我记住多少!

1、序列化本质其实就是 对象 –> 二进制,目的为了存储或者网络传输
2、序列化分为自动和手动;也就是实现两个不同的接口。
3、用户或者说开发者可以重写某些方法来控制序列化的过程
4、可以使用transient修饰符来控制某些字段不被序列化。但是这只适用于自动的
5、序列化代理模式能使序列化的安全性达到极高

疑惑一

ois.defaultReadObject();
oos.defaultWriteObject();

上面的作用是什么?

这是我在看继承那章节的代码时遇到的!我这里贴出完整代码:

首先假设,有这么一个父类

public class SuperClass {private int id;private String value;public int getId() {return id;}public void setId(int id) {this.id = id;}public String getValue() {return value;}public void setValue(String value) {this.value = value;}}

现在我们写个子类继承上面父类:

public class SubClass extends SuperClass implements Serializable, ObjectInputValidation{private static final long serialVersionUID = -1322322139926390329L;private String name;public String getName() {return name;}public void setName(String name) {this.name = name;}@Overridepublic String toString(){return "SubClass{id="+getId()+",value="+getValue()+",name="+getName()+"}";}//adding helper method for serialization to save/initialize super class stateprivate void readObject(ObjectInputStream ois) throws ClassNotFoundException, IOException{ois.defaultReadObject();//notice the order of read and write should be samesetId(ois.readInt());setValue((String) ois.readObject());}private void writeObject(ObjectOutputStream oos) throws IOException{oos.defaultWriteObject();oos.writeInt(getId());oos.writeObject(getValue());}@Overridepublic void validateObject() throws InvalidObjectException {//validate the object hereif(name == null || "".equals(name)) throw new InvalidObjectException("name can't be null or empty");if(getId() <=0) throw new InvalidObjectException("ID can't be negative or zero");}}

测试代码:

public class InheritanceSerializationTest {public static void main(String[] args) {String fileName = "subclass.ser";SubClass subClass = new SubClass();subClass.setId(10);subClass.setValue("Data");subClass.setName("Pankaj");try {SerializationUtil.serialize(subClass, fileName);} catch (IOException e) {e.printStackTrace();return;}try {SubClass subNew = (SubClass) SerializationUtil.deserialize(fileName);System.out.println("SubClass read = "+subNew);} catch (ClassNotFoundException | IOException e) {e.printStackTrace();}}}

结果为:

SubClass read = SubClass{id=10,value=Data,name=Pankaj}

此时我把下面代码注释掉:

//readObject方法中
ois.defaultReadObject();
//writeObject方法中
oos.defaultWriteObject();

其结果就变成了:

SubClass read = SubClass{id=10,value=Data,name=null}

这里可以猜测出,其作用:
因为子类是实现了序列化接口的,并且又重写了readObject()writeObject()方法。
那么在序列化或者反序列化时,就会走这两个方法,(并且不会再走ObjectInputStreamObjectOutputStream中的相应的方法)。那么defaultWriteObject()defaultReadObject()其实就是没重写时的默认方法。或者说,虽然我重写了,又不想改任何东西,那么就直接调这两个方法就可以了!

说白了,就是序列化的默认方法,对上面的子类而已,其默认只会序列化name这个字段,其从父类继承过来的,并不会序列化,所以你可以再重写的方法中手动处理,使其可以序列化。

后来我又做了个测试,既然子类就是要序列化name字段,我手动处理不也可以吗?
答案是肯定的:

private void readObject(ObjectInputStream ois) throws ClassNotFoundException, IOException {
//      ois.defaultReadObject();setName(ois.readUTF());setId(ois.readInt());setValue(ois.readObject().toString());
}private void writeObject(ObjectOutputStream oos) throws IOException{oos.writeUTF(getName());//oos.writeInt(getId());oos.writeObject(getValue());
}

也就是调用writeUTFreadUTF方法。这两个方法是以utf-8方式保存字符串。

readResolve与writeReplace作用或者说区别

这个是我在看最后一个例子时,注意到的;

最后一个例子是配合起来用,使序列化达到极高的安全状态。

writeReplace:这个用于序列化;java的序列化机制保证在序列化某个对象之前,先调用该对象的writeReplace()方法,如果该方法返回另一个java对象,则系统转为序列化另一个对象。

系统在序列化某个对象之前,会先调用该对象的writeReplace()和writeObject()两个方法,系统总是先调用被序列化对象的writeReplace()方法,如果该方法返回另一个对象,系统将再次调用另一个对象的writeReplace()方法,直到该方法不再返回另一个对象为止,程序最后将调用该对象的writeObject()方法来保存该对象的状态。

readResolve:用于反序列化;一般用于在序列化类中实现单例模式;

这个方法会紧挨着readObject()之后被调用,该方法的返回值将会代替原来反序列化的对象,而原来readObject()反序列化的对象将会立即丢弃。

readObject()方法在序列化单例类,枚举类时尤其有用。

个人觉得,其就是为了解决反序列化的单例问题而存在的。

我对最后一个例子做了个小测试:

private void readObject(ObjectInputStream ois) throws ClassNotFoundException, IOException{
//          ois.defaultReadObject();String oo = ois.readObject().toString();System.out.println(oo);if(oo.startsWith(PREFIX) && oo.endsWith(SUFFIX)){Data a = new Data("1");
a.setData( oo.substring(3, oo.length() -4));}
}/*private Object readResolve() throws InvalidObjectException {if(dataProxy.startsWith(PREFIX) && dataProxy.endsWith(SUFFIX)){return new Data(dataProxy.substring(3, dataProxy.length() -4));}else throw new InvalidObjectException("data corrupted");
}*/

上面注释掉的代码是其原来的代码;

我是改为重写readObject方法,但是报错了:

java.lang.ClassCastException: serializable.Data$DataProxy cannot be cast to serializable.Data

当然这个错,很明了,毕竟序列化保存在文件里的是代理类对象,那么反序列化得到的当然是代理类对象了;此时将其强转为Data,当然就错了;而我重写的那个方法,由于无法改变返回的对象,自然是没起到什么作用!所以需要使用readResolve方法来替换掉原来反序列化的对象(DataProxy)。

当然假设我们不强转的话,只得到代理类的话,还是OK的:
将测试代码修改下:

public static void main(String[] args) throws IOException, ClassNotFoundException {String fileName = "data.ser";Data data = new Data("Pankaj");SerializationUtil.serialize(data, fileName);Object newData =  SerializationUtil.deserialize(fileName);
System.out.println(newData);
}

其结果为:

ABCPankajDEFG
serializable.Data$DataProxy@119d7047

总结

目前大概是了解了写序列化的步骤;
之前还想着自己重写个类似功能的jar包出来。
现在看来,我之前的想法过于简单;

参考地址:

黑马程序员——————> 自定义序列化

readResolve()方法与序列化

Serializable:writeReplace

Java序列化示例教程

这篇关于java 序列化 父子类的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1112138

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S