一、序列化和反序列化的概念
序列化:把对象转换为字节序列的过程称为对象的序列化。
反序列化:把字节序列恢复为对象的过程称为对象的反序列化。
对象序列化的主要两种用途:
(1)把对象的字节序列永久地保存到硬盘上,通常放在一个文件中。
(2)在网络上传送对象的字节序列。
二、JDK类库中的序列化API
Java.io.ObjectOutputStream代表对象的输出流,它的writeObject(Object obj)方法可对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中。
Java.io.ObjectInputStream代表对象的输入流,它的readObject()方法从一个源输入流中读取字节序列,再把它们反序列化为一个对象,并将其返回。
只用实现了Serializable和Externalizable接口的类对想才能被序列化。Externalizable接口继承自Serializable接口,实现Externalizable接口的类完全由自身来控制序列化的行为,而仅实现Serializable接口的类可以采用默认的序列化方式。
对象序列化包括以下步骤:
(1)创建一个对象输出流,它可以包装一个其它类型的目标输出流,如文件输出流。
(2)通过对象输出流writeObject()方法写对象。
对象反序列化的步骤:
(1)创建一个对象输入流,它可以包装一个其它类型的源输入流,如文件输入流。
(2)通过对象输入流的readObject()方法读取对象。
定义一个Person类,实现Serializable接口:
import java.io.Serializable;
/*
* java.io.Serializable,该接口没有任何抽象方法,这种接口称为:签名接口
*虽然在源码中不需要重写方法,但实际编译器在编译该类为class文件时,会根据
*当前类结构添加一个方法,用于将当前类实例转换为一组字节。
*/
public class Person implements Serializable{
/**
* 序列化版本号,直接影响到序列化是否成功。
* 当一个类实现了Serializable接口后,编译器会提示,
* 我们应当添加一个常量:seriaVersionUID
* 序列化版本号影响者反序列化结果,当对象输入流在反序列化一个实例时,
* 会检查该实例与其所属的类的版本号是否一致,不一致则反序列化会抛出版本异常
* 若一致,则成功反序列化。
* 自行维护版本号可以主动确定反序列化结果,但是若不指定序列化版本号,
* 编译器在编译当前类时会根据当前类的结构生成一个版本号,但是只要当前类
* 发生改变,则版本号一定会变。
*/
private static final long serialVersionUID = 1L;
int age;
String name;
String sex;
//注意:transient在序列化时,忽略改属性,相应的,在反序列化时,也不存在该属性
private transient String[] otherInfo;
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public String[] getOtherInfo() {
return otherInfo;
}
public void setOtherInfo(String[] otherInfo) {
this.otherInfo = otherInfo;
}
}
序列化和反序列化Person类对象
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.text.MessageFormat;
public class TestObjSerializePerson {
//序列化对象
public static void SerializePerson() throws FileNotFoundException, IOException {
String[] otherInfo= {"我是大美人","!!!哈哈哈"};
Person person=new Person();
person.setAge(18);
person.setName("sqq");
person.setSex("女");
person.setOtherInfo(otherInfo);
ObjectOutputStream oos=new ObjectOutputStream(new FileOutputStream(new File("D:/Person.txt")));
/*
* java.io.ObjectOutputStream代表对象输出流,
* 它的writeObject(Object obj)方法可对参数指定的obj对象进行序列化,
* 把得到的字节序列写到一个目标输出流中。
*/
oos.writeObject(person);
System.out.println("对象序列化成功");
oos.close();
}
//反序列化对象
public static Person DeserializePerson() throws FileNotFoundException, IOException, ClassNotFoundException {
ObjectInputStream ois=new ObjectInputStream(new FileInputStream("D:/Person.txt"));
/*
* java.io.ObjectInputStream代表对象输入流,
* 它的readObject()方法从一个源输入流中读取字节序列,
* 再把它们反序列化为一个对象,并将其返回。
*/
Person person=(Person) ois.readObject();
System.out.println("反序列化成功");
return person;
}
public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException {
// TODO Auto-generated method stub
//序列化对象
SerializePerson();
//反序列化对象
Person p=DeserializePerson();
System.out.println(MessageFormat.format("name={0},age={1},sex={2},otherinfo={3}", p.getName(),p.getAge(),p.getSex(),p.getOtherInfo()));
}
}
三、serialVersionUID的取值
serialVersionUID的取值是Java运行时环境根据类的内部细节自动生成的。如果对类的源代码做了修改,在重新编译,新生成的类文件的serialVersionUID的取值也克宁会发生变化。
类的serialVersionUID的默认值完全依赖于Java编译器的实现,对于同一个类,用不同的Java编译器编译,有可能会导致不同的serialVerisonUID,也有可能相同。为了提高serialVersionUID的独立性和确定性,强烈建议在一个可序列化类中显示的定义serialVersionUID,为它赋予明确的值。
显示定义serialVersionUID有两种用途:
(1)在某些场合,希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有相同的serialVersionUID.
(2)在某些场合,不希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有不同serizlVersionUID;
转自:http://www.cnblogs.com/xdp-gacl/p/3777987.html