什么是序列化与反序列化?
Java的序列化是JDK1.1中引入的,其目的时为了将Java对象转换为字节数组,便于存储或传输。反序列化即将字节数组转换回Java对象保存时的状态。
解析
Java中的Serializable接口是一个空接口,没有任何实现。序列化的实际过程需要自己进行code,假设我们将对象写入到文件中,可以通过ObjectOutputStream将对象写入到文件中,然后可通过ObjectInputStream将对象从文件中读取出来。
如果我们只使用输入输出流,而没有实现Serializable接口,则会抛出异常。
其部分源码为:
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());
}
}
即ObjectOutputStream在进行序列化的时候回去判断对象属于哪个类型,然后才进行WriteObject。从代码可以看到,实现Serializable接口,会执行writeOrdinaryObject方法。
总结
Serializable接口,为空,它只是起到一个标识的作用,用来表明该对象可以序列化,真正的序列化和反序列化则不需要它来完成。
注意:static 和 transient 修饰的字段是不会被序列化的。
序列化保存的是对象的状态,而 static 修饰的字段属于类的状态,因此可以证明序列化并不保存 static 修饰的字段。
transient 的中文字义为“临时的”,它可以阻止字段被序列化到文件中,在被反序列化后,transient 字段的值被设为初始值,比如 int
型的初始值为 0,对象型的初始值为 null
。
序列化接口:Externalizable
该接口新增两个方法,writeExternal(),readExternal(),一个无参构造方法。
想要实现序列化,则首先是实现Externalizable接口,然后重写writeExternal(),readExternal()方法,否则只实现Externalizable接口,而不重写writeExternal(),readExternal(),则不能完后序列化。
⑴重写writeExternal()
只需将需要序列化对象的字段值写入输出流对象中即可
@Override
public void writeExternal(ObjectOutput out) throws IOException {
out.writeObject(name);
out.writeInt(age);
}
⑵重写readExternal()
只需将需要反序列化对象的字段从输入流对象中读取出即可
@Override
public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
name = (String) in.readObject();
age = in.readInt();
}
serialVersionUID
当实现序列化接口的时候,IDE会建议该类能生成一个序列化ID.它的作用是:
serialVersionUID 被称为序列化 ID,它是决定 Java 对象能否反序列化成功的重要因子。在反序列化时,Java 虚拟机会把字节流中的 serialVersionUID 与被序列化类中的 serialVersionUID 进行比较,如果相同则可以进行反序列化,否则就会抛出序列化版本不一致的异常。
如有问题,敬请指出,与君共勉;