ObjectOutputStream 将 Java 对象的基本数据类型和图形写入 OutputStream。可以使用 ObjectInputStream 读取(重构)对象。通过在流中使用文件可以实现对象的持久存储。如果流是网络套接字流,则可以在另一台主机上或另一个进程中重构对象。
只能将支持 java.io.Serializable 接口的对象写入流中。每个 serializable 对象的类都被编码,编码内容包括类名和类签名、对象的字段值和数组值,以及从初始对象中引用的其他所有对象的闭包。
简而言之就是可以通过ObjectOutputStream 把调用java.io.Serializable接口的类的对象写入流中,而在流的另一端取出这个对象来。也可以把对象写到文件里,以后再读出来。
实现串行化需要注意的条件:
1. 输入流写入的类的对象的方法与输出流的类的对象的方法必须名字、类型、修饰符完全一致,方法里面的内容可以不同。 (方法)(保证serialVersionUID一致)
2. 输入流的对象与输出流的对象字段必须一致,包括修饰符。 (字段)
3. 输入流的的对象的类与输出流对象的类必须一致,包括名字。 (类)
4. 不需要写入流的字段用transient修饰。
5. 串行化不会写入静态变量,成员方法不会被保存。
总而言之,串行化就是按格式传数据/读数据或写数据/读数据的。
如果想自定义数据传输的顺序,在实现Serializable接口的类中重写下面2种方法:
private void readObject(ObjectInputStream in)
{
}
private void writeObject(ObjectOutputStream out)
{
}
上面一个是写的方法,一个是读的方法。
下面是从源码中截取的一段:
if (obj instanceof String) {
writeString((String) obj, unshared);
} else if (cl.isArray()) {
writeArray(obj, desc, unshared);
} else if (obj instanceof Enum) {
writeEnum((Enum) obj, desc, unshared);
} else if (obj instanceof Serializable) {
writeOrdinaryObject(obj, desc, unshared);
} else {
if (extendedDebugInfo) {
throw new NotSerializableException(
cl.getName() + "\n" + debugInfoStack.toString());
} else {
throw new NotSerializableException(cl.getName());
}
}
Enum 常量的序列化不同于普通的 serializable 或 externalizable 对象。enum 常量的序列化形式只包含其名称;常量的字段值不被传送。为了序列化 enum 常量,ObjectOutputStream 需要写入由常量的名称方法返回的字符串。与其他 serializable 或 externalizable 对象一样,enum 常量可以作为序列化流中后续出现的 back 引用的目标。用于序列化 enum 常量的进程不可定制;在序列化期间,由 enum 类型定义的所有类特定的 writeObject 和 writeReplace 方法都将被忽略。类似地,任何 serialPersistentFields 或 serialVersionUID 字段声明也将被忽略,所有 enum 类型都有一个 0L 的固定的 serialVersionUID。