对象的输入输出流 : 主要的作用是用于写入对象信息与读取对象信息。 对象信息一旦写到文件上那么对象的信息就可以做到持久化了   对象的输出流ObjectOutputStream   对象的输入流:  ObjectInputStream

使用:

对象的输出流将指定的对象写入到文件的过程,就是将对象序列化的过程,对象的输入流将指定序列化好的文件读出来的过程,就是对象反序列化的过程。既然对象的输出流将对象写入到文件中称之为对象的序列化,那么可想而知对象所对应的class必须要实现Serializable接口。(查看源码可得知:Serializable接口没有任何的方法,只是作为一个标识接口存在)。

1、将User类的对象序列化

 1 class User implements Serializable{//必须实现Serializable接口
2 String uid;
3 String pwd;
4 public User(String _uid,String _pwd){
5 this.uid = _uid;
6 this.pwd = _pwd;
7 }
8 @Override
9 public String toString() {
10 return "账号:"+this.uid+" 密码:"+this.pwd;
11 }
12 }
13
14 public class Demo1 {
15
16 public static void main(String[] args) throws IOException {
17 //假设将对象信息写入到obj.txt文件中,事先已经在硬盘中建立了一个obj.txt文件
18 File f = new File("F:\\obj.txt");
19 writeObjec(f);
20 System.out.println("OK");
21 }
22
23 //定义方法把对象的信息写到硬盘上------>对象的序列化。
24 public static void writeObjec(File f) throws IOException{
25 FileOutputStream outputStream = new FileOutputStream(f);//创建文件字节输出流对象
26 ObjectOutputStream objectOutputStream = new ObjectOutputStream(outputStream);
27 objectOutputStream.writeObject(new User("酒香逢","123"));
28 //最后记得关闭资源,objectOutputStream.close()内部已经将outputStream对象资源释放了,所以只需要关闭objectOutputStream即可
29 objectOutputStream.close();
30 }
31 }

运行程序得到记事本中存入的信息:可见已经序列化到记事本中

2、将序列化到记事本的内容反序列化

 1 public class Demo1 {
2
3 public static void main(String[] args) throws IOException, ClassNotFoundException {
4 //假设将对象信息写入到obj.txt文件中,事先已经在硬盘中建立了一个obj.txt文件
5 File f = new File("F:\\obj.txt");
6 //writeObjec(f);
7 readObject(f);
8 System.out.println("OK");
9 }
10
11 //定义方法把对象的信息写到硬盘上------>对象的序列化。
12 public static void writeObjec(File f) throws IOException{
13 FileOutputStream outputStream = new FileOutputStream(f);//创建文件字节输出流对象
14 ObjectOutputStream objectOutputStream = new ObjectOutputStream(outputStream);
15 objectOutputStream.writeObject(new User("酒香逢","123"));
16 //最后记得关闭资源,objectOutputStream.close()内部已经将outputStream对象资源释放了,所以只需要关闭objectOutputStream即可
17 objectOutputStream.close();
18 }
19 //把文件中的对象信息读取出来-------->对象的反序列化
20 public static void readObject(File f) throws IOException, ClassNotFoundException{
21 FileInputStream inputStream = new FileInputStream(f);//创建文件字节输出流对象
22 ObjectInputStream objectInputStream = new ObjectInputStream(inputStream);
23 User user = (User)objectInputStream.readObject();
24 System.out.println(user);
25 }
26 }

运行代码得到的结果:

账号:酒香逢 密码:123 OK

但是,如果这时候这个obj.txt是我们项目中一个文件,而项目到后期在原来User类的基础上添加成员变量String userName;

 1 class User implements Serializable{//必须实现Serializable接口
2 String uid;
3 String pwd;
4 String userName="名字";//新添加的成员变量
5 public User(String _uid,String _pwd){
6 this.uid = _uid;
7 this.pwd = _pwd;
8 }
9 @Override
10 public String toString() {
11 return "账号:"+this.uid+" 密码:"+this.pwd;
12 }
13 }

这时候如果我们再反序列化,则会引发下面的异常:

Exception in thread "main" java.io.InvalidClassException: xuliehua.User; local class incompatible: stream classdesc serialVersionUID = 2161776237447595412, local class serialVersionUID = -3634244984882257127   at java.io.ObjectStreamClass.initNonProxy(ObjectStreamClass.java:604)   at java.io.ObjectInputStream.readNonProxyDesc(ObjectInputStream.java:1601)   at java.io.ObjectInputStream.readClassDesc(ObjectInputStream.java:1514)   at java.io.ObjectInputStream.readOrdinaryObject(ObjectInputStream.java:1750)   at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1347)   at java.io.ObjectInputStream.readObject(ObjectInputStream.java:369)   at xuliehua.Demo1.readObject(Demo1.java:48)   at xuliehua.Demo1.main(Demo1.java:32)

异常信息解读:

serialVersionUID 是用于记录class文件的版本信息的,serialVersionUID这个数字是JVM(JAVA虚拟界)通过一个类的类名、成员、包名、工程名算出的一个数字。而这时候序列化文件中记录的serialVersionUID与项目中的不一致,即找不到对应的类来反序列化。

3、如果序列化与反序列化的时候可能会修改类的成员,那么最好一开始就给这个类指定一个serialVersionUID,如果一类已经指定的serialVersionUID,然后 在序列化与反序列化的时候,jvm都不会再自己算这个 class的serialVersionUID了。

去掉刚才添加的成员变量userName;,并且在User类中指定一个serialVersionUID

 1 class User implements Serializable{//必须实现Serializable接口
2
3 private static final long serialVersionUID = 1L;
4 String uid;
5 String pwd;
6 //String userName="名字";//新添加的成员变量
7 public User(String _uid,String _pwd){
8 this.uid = _uid;
9 this.pwd = _pwd;
10 }
11 @Override
12 public String toString() {
13 return "账号:"+this.uid+" 密码:"+this.pwd;
14 }
15 }

重新序列化到obj.txt文件中,然后再类中再将userName添加回来(将上面User类中userName字段解注释),再一次执行反序列化操作,执行的结果跟之前反序列化的结果是一致的。可见这样解决后我们后期修改类也是可行的。

4、如果在User类中再添加成员变量,而这个变量为一个class ,如Address,那么Address类也必须要实现Serializable接口。

 1 class Address implements Serializable{
2 String country;
3 String city;
4 }
5
6 class User implements Serializable{//必须实现Serializable接口
7
8 private static final long serialVersionUID = 1L;
9 String uid;
10 String pwd;
11 String userName="名字";//新添加的成员变量
12 Address address;//成员变量为Address
13 public User(String _uid,String _pwd){
14 this.uid = _uid;
15 this.pwd = _pwd;
16 }
17 @Override
18 public String toString() {
19 return "账号:"+this.uid+" 密码:"+this.pwd;
20 }
21 }

5、最后再提一下关键字transient关键字,当你不想要某些字段序列化时候,可以用transient关键字修饰

 1 class User implements Serializable{//必须实现Serializable接口
2
3 private static final long serialVersionUID = 1L;
4 String uid;
5 String pwd;
6 transient String userName="名字";//新添加的成员变量//添加关键字transient后,序列化时忽略
7 Address address;//成员变量为Address
8 public User(String _uid,String _pwd){
9 this.uid = _uid;
10 this.pwd = _pwd;
11 }
12 @Override
13 public String toString() {
14 return "账号:"+this.uid+" 密码:"+this.pwd;
15 }
16 }

最后总结一下对象输入输出流使用时需要注意:

1. 如果对象需要被写出到文件上,那么对象所属的类必须要实现Serializable接口。 Serializable接口没有任何的方法,是一个标识接口而已。 2. 对象的反序列化创建对象的时候并不会调用到构造方法的、(这点文中没有说到,想要验证的同学在构造方法后面加一句System.out.println("构造方法执行吗?");,实际上构造方法是不执行的,自然这句话也没有输出了) 3. serialVersionUID 是用于记录class文件的版本信息的,serialVersionUID这个数字是通过一个类的类名、成员、包名、工程名算出的一个数字。 4. 使用ObjectInputStream反序列化的时候,ObjeectInputStream会先读取文件中的serialVersionUID,然后与本地的class文件的serialVersionUID 进行对比,如果这两个id不一致,反序列则失败。 5. 如果序列化与反序列化的时候可能会修改类的成员,那么最好一开始就给这个类指定一个serialVersionUID,如果一类已经指定的serialVersionUID,然后 在序列化与反序列化的时候,jvm都不会再自己算这个 class的serialVersionUID了。 6. 如果一个对象某个数据不想被序列化到硬盘上,可以使用关键字transient修饰。 7. 如果一个类维护了另外一个类的引用,则另外一个类也需要实现Serializable接口。

对象输入输出流ObjectInputStream、ObjectOutputStream(对象序列化与反序列化)的更多相关文章

  1. (JAVA)从零开始之--对象输入输出流ObjectInputStream、ObjectOutputStream(对象序列化与反序列化)

    对象的输入输出流 : 主要的作用是用于写入对象信息与读取对象信息. 对象信息一旦写到文件上那么对象的信息就可以做到持久化了 对象的输出流: ObjectOutputStream 对象的输入流:  Ob ...

  2. 对象输入输出流ObjectInputStream、ObjectOutputStream(对象的序列化与反序列化)

    如题 所有关联的类需要继承Serializable 接口 文件为空,直接反序列化为发生错误; 毕竟对象为null , 序列化到文件里不是空空的! 以下笔记的原文连接: https://www.cnbl ...

  3. 输入输出流ObjectInputStream、ObjectOutputStream(对象序列化与反序列化)

    对象的输入输出流 : 主要的作用是用于写入对象信息与读取对象信息. 对象信息一旦写到文件上那么对象的信息就可以做到持久化了 对象的输出流: ObjectOutputStream 对象的输入流:  Ob ...

  4. Day 18:SequenceInputStream、合并切割mp3、对象输入输出流对象

    SequenceInputStream用例题讲述用法 需求:1.把a.txt与b.txt 文件的内容合并 2.把a.txt与b.txt .c.txt文件的内容合并 import java.io.Fil ...

  5. serialVersionUID序列化版本号与ObjectOutputStream对象输入输出流

    1. 观察ObjectOutputStream 我们观察ObjectOutputStream就可以发现该类没有无参构造,只有有参构造,所以他是一个包装流 2. 具体使用: public static ...

  6. java 对象输入输出流

    对象的输入输出流的作用: 用于写入对象 的信息读取对象的信息. 对象的持久化. 比如:用户信息.              ObjectInputStream   : 对象输入流            ...

  7. (21)IO流之对象的序列化和反序列化流ObjectOutputStream和ObjectInputStream

    当创建对象时,程序运行时它就会存在,但是程序停止时,对象也就消失了.但是如果希望对象在程序不运行的情况下仍能存在并保存其信息,将会非常有用,对象将被重建并且拥有与程序上次运行时拥有的信息相同.可以使用 ...

  8. Java——IO流 对象的序列化和反序列化流ObjectOutputStream和ObjectInputStream

    对象的输入输出流 : 主要的作用是用于写入对象信息与读取对象信息. 对象信息一旦写到文件上那么对象的信息就可以做到持久化了 对象的输出流: ObjectOutputStream 对象的输入流:  Ob ...

  9. java 对象序列化与反序列化

    Java序列化与反序列化是什么? 为什么需要序列化与反序列化? 如何实现Java序列化与反序列化? 本文围绕这些问题进行了探讨. 1.Java序列化与反序列化  Java序列化是指把Java对象转换为 ...

随机推荐

  1. PyTorch 60 分钟入门教程:数据并行处理

    可选择:数据并行处理(文末有完整代码下载) 作者:Sung Kim 和 Jenny Kang 在这个教程中,我们将学习如何用 DataParallel 来使用多 GPU. 通过 PyTorch 使用多 ...

  2. <label>标签for属性

    label 元素不会向用户呈现任何特殊效果.不过,它为鼠标用户改进了可用性.如果您在 label 元素内点击文本,就会触发此控件.就是说,当用户选择该标签时,浏览器就会自动将焦点转到和标签相关的表单控 ...

  3. Python代码分析工具

    Python代码分析工具:PyChecker.Pylint - CSDN博客 https://blog.csdn.net/permike/article/details/51026156

  4. CXF 中自定义SOAPHeader

    Interceptor是CXF架构中一个很有特色的模式.你可以在不对核心模块进行修改的情况下,动态添加很多功能.这对于CXF这个以处理消息为中心的服务框架来说是非常有用的,CXF通过在Intercep ...

  5. LEA指令与MOV指令的区别——发现一本汇编好书

    一.汇编语言中PTR的含义及作用mov ax,bx ;是把BX寄存器“里”的值赋予AX,由于二者都是word型,所以没有必要加“WORD”mov ax,word ptr [bx];是把内存地址等于“B ...

  6. HDU1255 覆盖的面积 —— 求矩形交面积 线段树 + 扫描线 + 离散化

    题目链接:https://vjudge.net/problem/HDU-1255 给定平面上若干矩形,求出被这些矩形覆盖过至少两次的区域的面积. Input输入数据的第一行是一个正整数T(1<= ...

  7. 如何完成dedecms外部数据库调用|跨数据库数据调用

    第1步:打开网站include\taglib文件夹中找到sql.lib.php文件,并直接复制一些此文件出来,并把复制出来的这个文件重命名为mysql.lib.php. 注:mysql.lib.php ...

  8. 并不对劲的字符串专题(三):Trie树

    据说这些并不对劲的内容是<信息学奥赛一本通提高篇>的配套练习. 并不会讲Trie树. 1.poj1056-> 模板题. 2.bzoj1212-> 设dp[i]表示T长度为i的前 ...

  9. 分析函数Ratio_to_report使用

    分析函数Ratio_to_report( ) over()使用说明 表中需要计算单项占比:比如单项在部门占比多少,单项在公司占比多少.特别是在财务单项计算,部门个人薪水计算上. Ratio_to_re ...

  10. hdu 3503(有点小技巧的dfs(对结点加东西表示边的某些状态))

    Friends Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Sub ...