文章目录
前言
处理流:是对一个已存在的流进行处理和封装,通过所封装的流的功能调用实现对数据的操作。而处理流中也有不同的分类,此片介绍的是处理流中的对象流。
如果对处理流流有疑问,可查看此博客或自行百度!
Java之节点流和处理流
什么是对象流?
对象流用于存储和读取基本数据类型数据和对象的处理流。它的强大之处在于可以把java中的对象写入数据源中,也能把对象从数据源中还原回来。为了让对象持久化(把对象存储到本地),可以使用java的对象流处理对象,把对象的内容写到本地存储的文件中,也可以从本地文件中读取出来。
比如说:将100(int/整形类型)和== 你好(String/字符串类型)== 写入某文本,写入的时候需要保持文件的100为int类型,你好为String类型,此时就需要用到对象流。(在保持一个数据值的时候,能够把它的数据类型保存起来)
也就是常说的序列化和反序列化。
- 序列化:一个对象可以被表示为一个字节序列,该字节序列包括该对象的数据、有关对象的类型的信息和存储在对象中数据的类型(保存一个对象的时候,保存数据的值和数据类型就叫序列化)
- 反序列化:将序列化对象写入文件之后,可以从文件中读取出来(恢复数据的时候,恢复数据的值和数据类型)
如果需要让某个对象支持序列化支持,则必须让其类是可序列化的。
为了让某个类是可序列化,该类必须实现如下两个接口之一:
- Serializable:提供的通用数据保存和读取的接口(标记接口,没有方法)
- Externalizable:该接口有方法需要实现,一般使用Serializable接口实现
基本介绍
- ObjectOutputStream:提供序列化功能
- ObjectInputStream:提供反序列化功能
功能::提供了对基本类型或对象类型的序列化和反序列化的方法
根据图可知,这两个类提供的构造器中采用了修饰器模式(传入的是各自父类的子类的实现对象,用什么方式操作,就传什么)
ObjectOutputStream
把对象转成字节数据的输出到文件中保存,对象的输出过程称为序列化,可实现对象的持久存储。
构造器
具体操作方法可以查看JDK文档或者百度
JDK8英文在线文档
JDK8中文在线文档
使用ObjectOutputStream序列化基本数据类型和一个Dog对象(自定义),并保存到data.dat文件中
序列化后,保存的文件格式不是纯文本的,而是按照它的格式来保存
- Integer实现了Serializable接口(int自动装箱)
- Boolean实现了Serializable接口(boolean自动装箱)
- Character实现了Serializable接口(char自动装箱)
- Double实现了Serializable接口(double自动装箱)
- String实现了Serializable接口
注意:这里ObjectOutputStream_ 和Dog文件在同一目录下,如果不在同一个目录下,则需要导包!
importjava.io.FileOutputStream;importjava.io.IOException;importjava.io.ObjectOutputStream;/**
* 演示使用ObjectOutputStream的使用
*/publicclassObjectOutputStream_{publicstaticvoidmain(String[] args){String filePath ="E:\\data.dat";//ObjectOutputStream os =null;try{
os =newObjectOutputStream(newFileOutputStream(filePath));//序列化数据E:\\data.dat
os.writeInt(100);//int -> Integer(自动装箱)
os.writeBoolean(true);//boolean -> Boolean(自动装箱)
os.writeChar('a');//char -> Character
os.writeDouble(3.14);//double -> Double
os.writeUTF("hello");//String//保存自定义对象
os.writeObject(newDog("jack",4));}catch(IOException e){
e.printStackTrace();}finally{try{
os.close();//释放资源System.out.println("数据保存完毕(序列化)");}catch(IOException e){
e.printStackTrace();}}}}
- Dog类
importjava.io.Serializable;publicclassDogimplementsSerializable{privateString name;privateint age;publicDog(String name,int age){this.name = name;this.age = age;}publicStringgetName(){return name;}publicvoidsetName(String name){this.name = name;}publicintgetAge(){return age;}publicvoidsetAge(int age){this.age = age;}}
文件打开乱码属于正常现象,此代码执行之后文件中内容:
w d a@ 窺雲 hellosr com.chapter19.outputstream_.DogP戶嶏+/> I ageL namet Ljava/lang/String;xp t jack
ObjectInputStream
反序列化流,将之前使用 ObjectOutputStream 序列化的原始数据恢复为对象,以流的方式读取对象。
构造器
具体操作方法可以查看JDK文档或者百度
JDK8英文在线文档
JDK8中文在线文档
- 读取(反序列化)的顺序要和保存数据(序列化)的顺序一致,否则会出现异常。
恢复ObjectOutputStream输出到文件中的数据和类型,恢复到程序中(反序列化)
注意:自定义类读取时,需要将定义的类拷贝到工程(并且公有化)或者导包
- 这里我使用的是导包
importcom.chapter19.outputstream_.Dog;importjava.io.FileInputStream;importjava.io.IOException;importjava.io.ObjectInputStream;publicclassObjectInputStream_{publicstaticvoidmain(String[] args){String filePath ="E:\\data.dat";//指定反序列化的文件ObjectInputStream oi =null;try{
oi =newObjectInputStream(newFileInputStream(filePath));System.out.println(oi.readInt());System.out.println(oi.readBoolean());System.out.println(oi.readChar());System.out.println(oi.readDouble());System.out.println(oi.readUTF());Object obj = oi.readObject();System.out.println(obj);System.out.println("运行类型"+ obj.getClass());Dog o1 =(Dog) obj;System.out.println(o1.getName());}catch(IOException|ClassNotFoundException e){
e.printStackTrace();}finally{try{
oi.close();}catch(IOException e){
e.printStackTrace();}}}}
控制台输出如下:
a
3.14
hello
com.chapter19.outputstream_.Dog@6f496d9f
运行类型class com.chapter19.outputstream_.Dog
jack
对象处理流的使用细节
- 读写顺序要一致,如果顺序出错,则会抛出异常
- 要求序列化或反序列化对象,需要实现Serializable接口
- 序列化的类中建议添加SerialVersionUID,提高版本的兼容性 当文件中添加新的内容需要执行时,JVM会认为这个文件只是这个个版本的修改版或升级版,而不会认为是一个新的类(与游戏更新类似)
- 序列化对象时,默认将里面所有属性都进行序列化,但除了static或transient修饰的成员(被transient修饰的变量不参与序列化和反序列化)
- 序列化对象时,要求里面属性的类型也需要实现序列化接口(int、char、double等会自动封装类型)
- 序列化具有可继承性,也就是如果某类已经实现了序列化,则它的所有子类也已经默认实现了序列化
版权归原作者 墨辰JC 所有, 如有侵权,请联系我们删除。