序列化
JAVA程序运行过程中,我们可以在JVM堆内存区域中创建可复用的JAVA对象,程序运行过程中可以反复使用。
但是在一些场景下:写入磁盘文件或在网络上进行数据传输等,存储或传输的是二进制数据,估因此无法保留数据在JVM中对象的形态,因此序列化和反序列化就是解决此类问题的
序列化:将对象信息转化为二进制数据流的过程称为序列化
反序列化:将二进制数据流恢复成对象的过程,称为反序列化
Java序列化机制
java原生的序列化方式有两种
(1)使用Serializable接口
(2)使用Externalizable接口
使用Serializable接口进行序列化
定义一个实现 Serializable接口的类
@Data
@AllArgsConstructor
public class SerializableTest implements Serializable {
private static final long serialVersionUID = 1L;
private Long id;
private String name;
private static String staticTest;
}
(当前用例中使用了Lombok注解)
注:serialVersionUID的作用是,当序列化的时候的serialVersionUID与反序列化的时候的serialVersionUID不一致的时候,会跑出InvalidCalssException
1.实例化需要序列化的类,然后进行赋值,序列化
SerializableTest serializableTest = new SerializableTest(1L, "test");
SerializableTest.staticTest = "staticTest";
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("output.txt"));
objectOutputStream.writeObject(serializableTest);
objectOutputStream.close();
2.对刚才输出的文件进行反序列化
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("output.txt"));
SerializableTest serializableTest = (SerializableTest) objectInputStream.readObject();
objectInputStream.close();
System.out.println(serializableTest);
3.输出结果
SerializableTest(id=1, name=test)
注意:在上面的的用例中,SerializableTest类里有一个静态属性staticTest,对象序列化之前对此属性进行了赋值,但是反序列化时输出的对象中,并没有此属性。
原因是:静态变量不属于对象,而属于类,所以流里面就没有写入静态变量
使用Externalizable接口进行序列化
定义一个实现Externalizable接口的类
@Data
@AllArgsConstructor
@NoArgsConstructor
public class ExternalizableTest implements Externalizable {
private static final long serialVersionUID = 1L;
private Long id;
private String name;
@Override
public void writeExternal(ObjectOutput out) throws IOException {
out.writeLong(id);
out.writeUTF(name);
}
@Override
public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
this.id = in.readLong();
this.name = in.readUTF();
}
}
1.实例化需要序列化的类,然后进行赋值,序列化
ExternalizableTest serializableTest = new ExternalizableTest(1L, "test");
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("output.txt"));
serializableTest.writeExternal(objectOutputStream);
objectOutputStream.close();
2.对刚才输出的文件进行反序列化
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("output.txt"));
ExternalizableTest serializableTest = new ExternalizableTest();
serializableTest.readExternal(objectInputStream);
objectInputStream.close();
System.out.println(serializableTest);
3.输出结果
ExternalizableTest(id=1, name=test)
注意:这里反序列化时属性的取值顺序要和序列化时一致,否则同类型数据会错乱,不同类型数据会报错
两种序列化方式之间的取舍
个人认为,序列化一整个对象时,使用Serialzable接口更适合,想要灵活的自定义序列化,使用Externalizable更合适。虽然使用Serialzable序列化时可以通过transient关键字避免序列化某些属性,但是定制化不强,不能够灵活使用