正常情况下,一个类实现java序列化很简单,只需要implements Serializable接口即可,之后该类在跨jvm的传输过程中会遵照默认java序列化规则序列化和反序列化;不同jvm版本之间序列化方式稍有不同,但基本上都是兼容的。
在某些特殊情况下,可能需要自定义序列化和反序列化的行为,看下面例子:
Java代码
classAbstractSerializeDemo {
privateintx, y;
publicvoidinit(intx,inty) {
this.x = x;
this.y = y;
}
publicintgetX() {
returnx;
}
publicintgetY() {
returny;
}
publicvoidprintXY() {
System.out.println("x:"+ x +";y:"+ y);
}
}
publicclassSerializeDemoextendsAbstractSerializeDemoimplementsSerializable {
privateintz;
publicSerializeDemo() {
super.init(10,50);
z =100;
}
publicvoidprintZ() {
super.printXY();
System.out.println("z:"+ z);
}
publicstaticvoidmain(String[] args)throwsIOException, ClassNotFoundException {
ByteArrayOutputStream bos =newByteArrayOutputStream();
ObjectOutputStream out =newObjectOutputStream(bos);
SerializeDemo sd =newSerializeDemo();
sd.printZ();
out.writeObject(sd);
ObjectInputStream in =newObjectInputStream(newByteArrayInputStream(bos.toByteArray()));
SerializeDemo sd2 = (SerializeDemo) in.readObject();
sd2.printZ();
}
}
这段程序表示了一个可序列化的类继承自一个非序列化的有状态超类,期望的结果是,子类序列化以后传输并反序列化回来,原先的值域包括超类的值域都保持不变。
但是输出是:
Java代码
x:10;y:50
z:100
x:0;y:0
z:100
结果和期望不符,子类的值域保留下来了,但是超类的值域丢失了,这对jvm来说是正常的,因为超类不可序列化;
为了解决这个问题,只能自定义序列化行为,具体做法是在SerializeDemo里加入以下代码:
Java代码
privatevoidwriteObject(ObjectOutputStream os)throwsIOException {
os.defaultWriteObject();//java对象序列化默认操作
os.writeInt(getX());
os.writeInt(getY());
}
privatevoidreadObject(ObjectInputStream is)throwsIOException, ClassNotFoundException {
is.defaultReadObject();//java对象反序列化默认操作
intx=is.readInt();
inty=is.readInt();
super.init(x,y);
}
writeObject和readObject方法为JVM会在序列化和反序列化java对象时会分别调用的两个方法,修饰符都是private,没错。
我们在序列化的默认动作之后将超类里的两个值域x和y也写入object流;与之对应在反序列化的默认操作之后读入x和y两个值,然后调用超类的初始化方法。
再次执行程序之后的输出为:
Java代码
x:10;y:50
z:100
x:10;y:50
z:100
另外还有两个自定义序列化方法writeReplace和readResolve,分别用来在序列化之前替换序列化对象 和 在反序列化之后的对返回对象的处理。一般可以用来避免singleTon对象跨jvm序列化和反序列化时产生多个对象实例,事实上singleTon的对象一旦可序列化,它就不能保证singleTon了。JVM的Enum实现里就是重写了readResolve方法,由JVM保证Enum的值都是singleTon的,所以建议多使用Enum代替使用writeReplace和readResolve方法。
Java代码
privateObject readResolve()
{
returnINSTANCE;
}
privateObject writeReplace(){
returnINSTANCE;
}
注:writeReplace调用在writeObject前;readResolve调用在readObject之后。
【编辑推荐】
【责任编辑:金贺 TEL:(010)68476606】
点赞 0