Java 提供了一种对象序列化的机制,该机制中,一个对象可以被表示为一个字节序列,该字节序列包括该对象的数据、有关对象的类型的信息和存储在对象中数据的类型。 将序列化对象写入文件之后,可以从文件中读取出来,并且对它进行反序列化,也就是说,对象的类型信息、对象的数据,还有对象中的数据类型可以用来在内存中新建对象。 整个过程都是 Java 虚拟机(JVM)独立的,也就是说,在一个平台上序列化的对象可以在另一个完全不同的平台上反序列化该对象。 类 ObjectInputStream 和 ObjectOutputStream 是高层次的数据流,它们包含反序列化和序列化对象的方法。
ObjectOutputStream 类包含很多写方法来写各种数据类型,但是一个特别的方法例外:public final void writeObject(Object x) throws IOException
上面的方法序列化一个对象,并将它发送到输出流。相似的ObjectInputStream 类包含如下反序列化一个对象的方法:public final Object readObject() throws IOException, ClassNotFoundException 该方法从流中取出下一个对象,并将对象反序列化。它的返回值为Object,因此,你需要将它转换成合适的数据类型。
请注意,一个类的对象要想序列化成功,必须满足两个条件: 该类必须实现 java.io.Serializable 接口。 该类的所有属性必须是可序列化的。 如果有一个属性不是可序列化的,则该属性必须注明是短暂的。 如果你想知道一个 Java标准类是否是可序列化的,请查看该类的文档。检验一个类的实例是否能序列化十分简单, 只需要查看该类有没有实现 java.io.Serializable接口。
ObjectOutputStream 类用来序列化一个对象,如下的例子实例化了一个 Hero对象,并将该对象序列化到一个文件中。 该程序执行后,就创建了一个名为 Hero.ser 文件。该程序没有任何输出,但是你可以通过代码研读来理解程序的作用。 注意: 当序列化一个对象到文件时, 按照 Java 的标准约定是给文件一个 .ser 扩展名。
package day9.work1; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.io.Serializable; class Hero implements Serializable {// 实现序列化接口 String name; int gk; public Hero(String name, int gk) { super(); this.name = name; this.gk = gk; } } public class Test7 { public static void main(String[] args) { // TODO Auto-generated method stub // 序列化对象 Hero h = new Hero("zs", 10); try { FileOutputStream fos = new FileOutputStream("E:/a.ser"); ObjectOutputStream bos = new ObjectOutputStream(fos); bos.writeObject(h); bos.close(); fos.close(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } // 反序列化对象 try { FileInputStream fos = new FileInputStream("E:/a.ser"); ObjectInputStream in = new ObjectInputStream(fos); Hero he = (Hero) in.readObject();// 类名一定要大写 System.out.println("姓名" + he.name + "级别" + he.gk); in.close(); fos.close(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } } } 姓名zs级别10这里要注意以下要点: readObject() 方法中的 try/catch代码块尝试捕获 ClassNotFoundException 异常。对于 JVM 可以反序列化对象,它必须是能够找到字节码的类。如果JVM在反序列化对象的过程中找不到该类,则抛出一个 ClassNotFoundException 异常。 注意,readObject() 方法的返回值被转化成 Hero引用。 当对象被序列化时,属性 SSN 的值为 111222333,但是因为该属性是短暂的,该值没有被发送到输出流。所以反序列化后 Hero 对象的 SSN 属性为 0。
一般情况下,我们在定义实体类时会继承Serializable接口
一个对象序列化的接口,一个类只有实现了Serializable接口,它的对象才能被序列化。
序列化是将对象状态转换为可保持或传输的格式的过程。与序列化相对的是反序列化,它将流转换为对象。这两个过程结合起来,可以轻松地存储和传输数据。
把对象转换为字节序列的过程称为对象的序列化 把字节序列恢复为对象的过程称为对象的反序列化
当我们需要把对象的状态信息通过网络进行传输,或者需要将对象的状态信息持久化,以便将来使用时都需要把对象进行序列化
那为什么还要继承Serializable。那是存储对象在存储介质中,以便在下次使用的时候,可以很快捷的重建一个副本。
或许你会问,我在开发过程中,实体并没有实现序列化,但我同样可以将数据保存到mysql、Oracle数据库中,为什么非要序列化才能存储呢?
我们来看看Serializable到底是什么,跟进去看一下,我们发现Serializable接口里面竟然什么都没有,只是个空接口 一个接口里面什么内容都没有,我们可以将它理解成一个标识接口。 在Java中的这个Serializable接口其实是给jvm看的,通知jvm,我不对这个类做序列化了,你(jvm)帮我序列化就好了。
Serializable接口就是Java提供用来进行高效率的异地共享实例对象的机制,实现这个接口即可。
JVM是Java Virtual Machine(Java虚拟机)的缩写,JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的。