上一篇Serializable接口,描述了序列化和反序列化机制,其操作均是基于ObjectInputStream和ObjectOutputStream。
根据ObjectInputStream和ObjectOutputStrema类结构,需先了解两个接口:
DataInput和DataOutput
DataInput和DataOutput 用于从字节流中读取/写入Java基本类型数据或UTF编码的字符串,比较特殊的是,到达流末尾时通过抛出EOFException的方式来表示的
ObjectInput和ObjectOutput分别扩展了DataInput和DataOutput:
ObjectInput扩展了DataInput,使其支持读取对象,并缓存到一个字节数组中
ObjectOutput扩展了DataOutpu,使其可以直接将对象写入流,货主写入字节数组中
ObjectInputStream和ObjectOutputStream是比较复杂,这次主要是学习其在序列化和反序列化上的使用,以及通过readObject和writeObject实现自定义序列化的原理。
在实例化ObjectInputStream和ObjectOutputStream时,均是通过其有参构造函数实例化,传递一个FileInputStream或FileOutputStream对象。
ObjectInputStream的readObject方法:
public final Object readObject() throws IOException, ClassNotFoundException {
if(enableOverrid) {
return readObjectOverride();
}
int outerHandle = passHandle;
try {
Object obj = readObject0(false);
handles.markDependency(outerHandle, passHandle);
ClassNotFoundException ex = handles.lookupException(passHandle);
if (ex != null) {
throw ex;
}
if (depth == 0) {
vlist.doCallback();
}
return obj;
} finally {
passHandle = outerHandle;
if (closed && depth == 0) {
clear();
}
}
}
readObject方法是final的,表示不可被Override,但是其已经通过enableOverrid标记实现了扩展,若想自定义实现,可以Override这个方法:
protected Object readObjectOverride() throws IOException, ClassNotFoundException {
return null;
}
该方式非常精妙,从另一种角度,实现了final修饰的方法的Override!而且将真正的处理方法封装到readObject0中
readObject方法核心的就是调用了readObject0方法:
try {
switch (tc) {
case TC_NULL:
return readNull();
case TC_REFERENCE:
return readHandle(unshared);
case TC_CLASS:
return readClass(unshared);
case TC_CLASSDESC:
case TC_PROXYCLASSDESC:
return readClassDesc(unshared);
case TC_STRING:
case TC_LONGSTRING:
return checkResolve(readString(unshared));
case TC_ARRAY:
return checkResolve(readArray(unshared));
case TC_ENUM:
return checkResolve(readEnum(unshared));
case