处理流:是对一个已存在的流进行处理和封装,通过所封装的流的功能调用实现对数据的操作。而处理流中也有不同的分类,此片介绍的是处理流中的对象流。
如果对处理流流有疑问,可查看此博客或自行百度!
Java之节点流和处理流
对象流用于存储和读取基本数据类型数据和对象的处理流。它的强大之处在于可以把java中的对象写入数据源中,也能把对象从数据源中还原回来。为了让对象持久化(把对象存储到本地),可以使用java的对象流处理对象,把对象的内容写到本地存储的文件中,也可以从本地文件中读取出来。
比如说:将100(int/整形类型)和== 你好(String/字符串类型)== 写入某文本,写入的时候需要保持文件的100为int类型,你好为String类型,此时就需要用到对象流。(在保持一个数据值的时候,能够把它的数据类型保存起来)
也就是常说的序列化和反序列化。
如果需要让某个对象支持序列化支持,则必须让其类是可序列化的。
为了让某个类是可序列化,该类必须实现如下两个接口之一:
- Serializable:提供的通用数据保存和读取的接口(标记接口,没有方法)
- Externalizable:该接口有方法需要实现,一般使用Serializable接口实现
功能::提供了对基本类型或对象类型的序列化和反序列化的方法
根据图可知,这两个类提供的构造器中采用了修饰器模式(传入的是各自父类的子类的实现对象,用什么方式操作,就传什么)
把对象转成字节数据的输出到文件中保存,对象的输出过程称为序列化,可实现对象的持久存储。
构造器
具体操作方法可以查看JDK文档或者百度
JDK8英文在线文档
JDK8中文在线文档
使用ObjectOutputStream序列化基本数据类型和一个Dog对象(自定义),并保存到data.dat文件中
序列化后,保存的文件格式不是纯文本的,而是按照它的格式来保存
注意:这里ObjectOutputStream_ 和Dog文件在同一目录下,如果不在同一个目录下,则需要导包!
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutputStream;/*** 演示使用ObjectOutputStream的使用*/
public class ObjectOutputStream_ {public static void main(String[] args) {String filePath = "E:\\data.dat";//ObjectOutputStream os = null;try {os = new ObjectOutputStream(new FileOutputStream(filePath));//序列化数据E:\\data.datos.writeInt(100);//int -> Integer(自动装箱)os.writeBoolean(true);//boolean -> Boolean(自动装箱)os.writeChar('a');//char -> Characteros.writeDouble(3.14);//double -> Doubleos.writeUTF("hello");//String//保存自定义对象os.writeObject(new Dog("jack", 4));} catch (IOException e) {e.printStackTrace();} finally {try {os.close();//释放资源System.out.println("数据保存完毕(序列化)");} catch (IOException e) {e.printStackTrace();}}}
}
import java.io.Serializable;public class Dog implements Serializable {private String name;private int age;public Dog(String name, int age) {this.name = name;this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}
}
文件打开乱码属于正常现象,此代码执行之后文件中内容:
w d a@ 窺雲 hellosr com.chapter19.outputstream_.DogP戶嶏+/> I ageL namet Ljava/lang/String;xp t jack
反序列化流,将之前使用 ObjectOutputStream 序列化的原始数据恢复为对象,以流的方式读取对象。
构造器
具体操作方法可以查看JDK文档或者百度
JDK8英文在线文档
JDK8中文在线文档
恢复ObjectOutputStream输出到文件中的数据和类型,恢复到程序中(反序列化)
注意:自定义类读取时,需要将定义的类拷贝到工程(并且公有化)或者导包
import com.chapter19.outputstream_.Dog;import java.io.FileInputStream;
import java.io.IOException;
import java.io.ObjectInputStream;public class ObjectInputStream_ {public static void main(String[] args) {String filePath = "E:\\data.dat";//指定反序列化的文件ObjectInputStream oi = null;try {oi = new ObjectInputStream(new FileInputStream(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等会自动封装类型)
序列化具有可继承性,也就是如果某类已经实现了序列化,则它的所有子类也已经默认实现了序列化
上一篇:力扣sql简单篇练习(二)
下一篇:DPDK系列之一基础环境搭建