对象序列化

对象序列化的目标是将对象保存到磁盘中,或允许在网络中直接传输对象。对象序列化机制允许把内存中的 Java 对象转换成平台无关的二进制流,从而允许把这种二进制流持久地保存在磁盘上,通过网络将这种二进制流传输到另1个网络节点。其他程序一旦获得了这种二进制流(无论是从磁盘中获取的,还是通过网络获取的) ,都可以将这种二进制流恢复成原来的 Java 对象

如何来使用对象序列化?

前提:

首先,一个类要被序列化,那么它必须实现如下两个接口之一:
  • Serializable
  • Extemalizable
实现着两个接口的作用,其实就是就是标记一个对象是否可以被序列化。所以比如我们有一个类需要被序列化,我们实现Serializable接口就行,无需实现任何方法

序列化工具类:ObjectOutputStream

方法:

  • writeObject()

反序列化工具类:objectInputStream 

方法:

  • readObject()
示例:存储Person类对象
import java.io.*;

/**
* @ClassName ObjectOutputStreamWriteObjectExample
* @projectName: object1
* @author: Zhangmingda
* @description: Person 类实现序列化接口
* date: 2021/4/18.
*/
public class ObjectOutputStreamWriteObjectExample {
//实现接口Serializable可以序列化
private static class Person implements Serializable {
String name;
int age; public Person(String name, int age) {
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
//序列化方法
private static void writeObject(Person person,String dstPath) {
try(ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream(dstPath))){
objectOutputStream.writeObject(person);
} catch (IOException e) {
e.printStackTrace();
}
}
//反序列化方法
private static<T> void readObject(String srcPath){
try(ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(srcPath))){
T person = (T) objectInputStream.readObject();
System.out.println(person.getClass());
System.out.println(person);
/**
* class ObjectOutputStreamWriteObjectExample$Person
* Person{name='钱老大', age=30}
*/
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
Person person = new Person("钱老大",30);
String dstPath = "输入输出文件读写/src/test/output/object-save.txt";
writeObject(person,dstPath);
String srcPath = "输入输出文件读写/src/test/output/object-save.txt";
readObject(srcPath);
}
}
示例:带有引用变量的对象序列化
引用类型也必须是可以序列化的。
import java.io.*;

/**
* @ClassName ObjectOutputStreamWriteObjectExample
* @projectName: object1
* @author: Zhangmingda
* @description: Person 类实现序列化接口
* date: 2021/4/18.
*/
public class ObjectOutputStreamWriteObjectExample {
//实现接口Serializable可以序列化
private static class Person implements Serializable {
String name;
int age;
Job job; public Person(String name, int age, Job job) {
this.name = name;
this.age = age;
this.job = job;
} public Person(String name, int age) {
this.name = name;
this.age = age;
} @Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
", job=" + job.getName() +
'}';
}
}
private static class Job implements Serializable {
private String name; public Job(String name) {
this.name = name;
}
public String getName() {
return name;
}
} //序列化方法
private static void writeObject(Person person,String dstPath) {
try(ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream(dstPath))){
objectOutputStream.writeObject(person);
} catch (IOException e) {
e.printStackTrace();
}
}
//反序列化方法
private static<T> void readObject(String srcPath){
try(ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(srcPath))){
T person = (T) objectInputStream.readObject();
System.out.println(person.getClass());
System.out.println(person);
/**
* class ObjectOutputStreamWriteObjectExample$Person
* Person{name='钱老大', age=30, job=教师}
*/
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
Person person = new Person("钱老大",30,new Job("教师"));
String dstPath = "输入输出文件读写/src/test/output/object-save.txt";
writeObject(person,dstPath);
String srcPath = "输入输出文件读写/src/test/output/object-save.txt";
readObject(srcPath);
}
}

指定哪些字段不需要序列化

比如上面的Person对象,可能有的女生认为年龄是一个秘密,不应该被序列化,怎么办?
transient关键字修饰对应的字段就OK了
示例:private transient int age;

存入:Person{name='钱老大', age=30, job=教师}

读取:Person{name='钱老大', age=0, job=教师} 【不被反序列化的字段置为 对应数据类型默认值

//序列化写入到文件时transient 未设置,读取时添加了transient,等于对象的类在写入后,读取之前又发生了变化,然后反序列化读取发生
错误类似:local class incompatible: stream classdesc serialVersionUID ......

解决:类在创建时 添加serialVersionUID字段来保证类发生变化后,也能反序列化成功(不被反序列化的字段置为 对应数据类型默认值)

即:类的升级

import java.io.*;

/**
* @ClassName ObjectOutputStreamWriteObjectExample
* @projectName: object1
* @author: Zhangmingda
* @description: Person 类实现序列化接口
* date: 2021/4/18.
*/
public class ObjectOutputStreamWriteObjectExample {
//实现接口Serializable可以序列化
private static class Person implements Serializable {
private static final long serialVersionUID = 769204331117184540L;
private String name;
private transient int age;
private Job job; public Person(String name, int age, Job job) {
this.name = name;
this.age = age;
this.job = job;
} public Person(String name, int age) {
this.name = name;
this.age = age;
} @Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
", job=" + job.getName() +
'}';
}
}
private static class Job implements Serializable {
private String name; public Job(String name) {
this.name = name;
}
public String getName() {
return name;
}
} //序列化方法
private static void writeObject(Person person,String dstPath) {
try(ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream(dstPath))){
objectOutputStream.writeObject(person);
} catch (IOException e) {
e.printStackTrace();
}
}
//反序列化方法
private static<T> void readObject(String srcPath){
try(ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(srcPath))){
T person = (T) objectInputStream.readObject();
System.out.println(person.getClass());
System.out.println(person);
/**
* class ObjectOutputStreamWriteObjectExample$Person
* Person{name='钱老大', age=30, job=教师}
*/
} catch (IOException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
// Person person = new Person("钱老大",30,new Job("教师"));
// String dstPath = "输入输出文件读写/src/test/output/object-save.txt";
// writeObject(person,dstPath);
String srcPath = "输入输出文件读写/src/test/output/object-save.txt";
readObject(srcPath);
}
}
 
 
 
 
 
 
 

java 输入输出 对象序列化implements Serializable与反序列化:ObjectOutputStream.writeObject() ;objectInputStream.readObject() ;serialVersionUID字段注意的更多相关文章

  1. Java Io 对象序列化和反序列化

    Java 支持将任何对象进行序列化操作,序列化后的对象文件便可通过流进行网络传输. 1.      对象序列化就是将对象转换成字节序列,反之叫对象的反序列化 2.      序列化流ObjectOut ...

  2. 【java】对象序列化Serializable、transient

    package 对象序列化; import java.io.Serializable; @SuppressWarnings("serial") class A implements ...

  3. Java之对象序列化和反序列化

    一.对象序列化和反序列化存在的意义: 当你创建对象,只要你需要,他就一直存在,但当程序结束,对象就会消失,但是存在某种情况,如何让程序在不允许的状态,仍然保持该对象的信息.并在下次程序运行的时候使用该 ...

  4. Java程序设计——对象序列化

    对象序列化的目标是将对象保存到磁盘中或允许在网络中直接传输对象,对象序列化机制允许把内存中的Java对象转换成平台无关的二进制流,从而允许把这种二进制流持久保存在磁盘上,通过网络将这种二进制流传输到另 ...

  5. 谈谈 JAVA 的对象序列化

    所谓的『JAVA 对象序列化』就是指,将一个 JAVA 对象所描述的所有内容以文件 IO 的方式写入二进制文件的一个过程.关于序列化,主要涉及两个流,ObjectInputStream 和 Objec ...

  6. java对象序列化byte[] and byte[]反序列化对象--转

    import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOExceptio ...

  7. js对象序列化JSON.stringify()与反序列化JSON.parse()

    草稿: parse用于从一个字符串中解析出json对象,如 var str = '{"name":"huangxiaojian","age" ...

  8. JAVA基础之——序列化和反序列化

    1 概念 序列化,将java对象转换成字节序列的过程. 反序列化,将字节序列恢复成java对象的过程. 2 为什么要序列化? 2.1 实现数据持久化,当对象创建后,它就会一直在,但是在程序终止时,这个 ...

  9. Java 序列化 对象序列化和反序列化

    Java 序列化 对象序列化和反序列化 @author ixenos 对象序列化是什么 1.对象序列化就是把一个对象的状态转化成一个字节流. 我们可以把这样的字节流存储为一个文件,作为对这个对象的复制 ...

随机推荐

  1. CF1559D2 Mocha and Diana (Hard Version)

    考虑到加树边每次最多只导致一对联通块之间的状态. 所以我们以任意顺序加入当前的合法边. 我们考虑先加入所有可加的\((1,a)\) 然后统计只在\(A\)中与1连的点,\(B\)中与2连的点. 则他们 ...

  2. 洛谷 P5643 - [PKUWC2018]随机游走(Min-Max 容斥+FWT+树上高斯消元,hot tea)

    题面传送门 一道挺综合的 hot tea,放到 PKUWC 的 D2T2 还挺喜闻乐见的( 首先我们考虑怎样对一个固定的集合 \(S\) 计算答案,注意到我们要求的是一个形如 \(E(\max(S)) ...

  3. 【每天五分钟大数据-第一期】 伪分布式+Hadoopstreaming

    说在前面 之前一段时间想着把 LeetCode 每个专题完结之后,就开始着手大数据和算法的内容. 想来想去,还是应该穿插着一起做起来. 毕竟,如果只写一类的话,如果遇到其他方面,一定会遗漏一些重要的点 ...

  4. flink---实时项目--day01--1. openrestry的安装 2. 使用nginx+lua将日志数据写入指定文件中 3. 使用flume将本地磁盘中的日志数据采集到的kafka中去

    1. openrestry的安装 OpenResty = Nginx + Lua,是⼀一个增强的Nginx,可以编写lua脚本实现⾮非常灵活的逻辑 (1)安装开发库依赖 yum install -y ...

  5. 大数据学习day23-----spark06--------1. Spark执行流程(知识补充:RDD的依赖关系)2. Repartition和coalesce算子的区别 3.触发多次actions时,速度不一样 4. RDD的深入理解(错误例子,RDD数据是如何获取的)5 购物的相关计算

    1. Spark执行流程 知识补充:RDD的依赖关系 RDD的依赖关系分为两类:窄依赖(Narrow Dependency)和宽依赖(Shuffle Dependency) (1)窄依赖 窄依赖指的是 ...

  6. cookie规范(RFC6265)翻译

    来源:https://github.com/renaesop/blog/issues/4 RFC 6265 要点翻译 1.简介 本文档定义了HTTP Cookie以及HTTP头的Set-Cookie字 ...

  7. nodejs-npm模块管理器

    JavaScript 标准参考教程(alpha) 草稿二:Node.js npm模块管理器 GitHub TOP npm模块管理器 来自<JavaScript 标准参考教程(alpha)> ...

  8. 30个类手写Spring核心原理之环境准备(1)

    本文节选自<Spring 5核心原理> 1 IDEA集成Lombok插件 1.1 安装插件 IntelliJ IDEA是一款非常优秀的集成开发工具,功能强大,而且插件众多.Lombok是开 ...

  9. 【Android】我有放入Icon到mipmap,但不显示,只显示安卓机器人Icon(Android 8.0 图标适配)

    首先,放上别人写的博客,而我自己的博客,只会写大概思路,给自己留给备忘 https://blog.csdn.net/guolin_blog/article/details/79417483 其实会发生 ...

  10. Linux学习 - 使用qq邮箱发送邮件

    1 打开qq邮箱,设置->账户->POP3/SMTP,开启服务 2 配置/etc/mail.rc文件 set from=73***32@qq.com #设置发送方邮件地址 set smtp ...