最近公司的在做服务化, 需要把所有 model 包里的类都实现 Serializable 接口, 同时还要显示指定 serialVersionUID 的值. 听到这个需求, 我脑海里就突然出现了好几个问题, 比如说:

  • 序列化和反序列化是什么?
  • 实现序列化和反序列化为什么要实现 Serializable 接口?
  • 实现 Serializable 接口就算了, 为什么还要显示指定 serialVersionUID 的值?
  • 我要为 serialVersionUID 指定个什么值?

下面我们来一一解答这几个问题.

序列化和反序列化

  • 序列化:把对象转换为字节序列的过程称为对象的序列化.
  • 反序列化:把字节序列恢复为对象的过程称为对象的反序列化.

    什么时候需要用到序列化和反序列化呢?

    当我们只在本地 JVM 里运行下 Java 实例, 这个时候是不需要什么序列化和反序列化的, 但当我们需要将内存中的对象持久化到磁盘, 数据库中时, 当我们需要与浏览器进行交互时, 当我们需要实现 RPC 时, 这个时候就需要序列化和反序列化了.
    前两个需要用到序列化和反序列化的场景, 是不是让我们有一个很大的疑问? 我们在与浏览器交互时, 还有将内存中的对象持久化到数据库中时, 好像都没有去进行序列化和反序列化, 因为我们都没有实现 Serializable 接口, 但一直正常运行.
    下面先给出结论:
    只要我们对内存中的对象进行持久化或网络传输, 这个时候都需要序列化和反序列化.
    理由:
    服务器与浏览器交互时真的没有用到 Serializable 接口吗? JSON 格式实际上就是将一个对象转化为字符串, 所以服务器与浏览器交互时的数据格式其实是字符串, 我们来看来 String 类型的源码:

    1. public final class String
    2. implements java.io.Serializable, Comparable<String>, CharSequence {
    3. /\*\* The value is used for character storage. \*/
    4. private final char value\[\];
    5. /\*\* Cache the hash code for the string \*/
    6. private int hash; // Default to 0
    7. /\*\* use serialVersionUID from JDK 1.0.2 for interoperability \*/
    8. private static final long serialVersionUID = -6849794470754667710L;
    9. ......
    10. }

    String 类型实现了 Serializable 接口, 并显示指定 serialVersionUID 的值.
    然后我们再来看对象持久化到数据库中时的情况, Mybatis 数据库映射文件里的 insert 代码:

    1. <insert id="insertUser" parameterType="org.tyshawn.bean.User">
    2. INSERT INTO t\_user(name, age) VALUES (#{name}, #{age})
    3. </insert>

    实际上我们并不是将整个对象持久化到数据库中, 而是将对象中的属性持久化到数据库中, 而这些属性都是实现了 Serializable 接口的基本属性.

    实现序列化和反序列化为什么要实现 Serializable 接口?

    在 Java 中实现了 Serializable 接口后, JVM 会在底层帮我们实现序列化和反序列化, 如果我们不实现 Serializable 接口, 那自己去写一套序列化和反序列化代码也行, 至于具体怎么写, Google 一下你就知道了.

    实现 Serializable 接口就算了, 为什么还要显示指定 serialVersionUID 的值?

    如果不显示指定 serialVersionUID, JVM 在序列化时会根据属性自动生成一个 serialVersionUID, 然后与属性一起序列化, 再进行持久化或网络传输. 在反序列化时, JVM 会再根据属性自动生成一个新版 serialVersionUID, 然后将这个新版 serialVersionUID 与序列化时生成的旧版 serialVersionUID 进行比较, 如果相同则反序列化成功, 否则报错.
    如果显示指定了 serialVersionUID, JVM 在序列化和反序列化时仍然都会生成一个 serialVersionUID, 但值为我们显示指定的值, 这样在反序列化时新旧版本的 serialVersionUID 就一致了.
    在实际开发中, 不显示指定 serialVersionUID 的情况会导致什么问题? 如果我们的类写完后不再修改, 那当然不会有问题, 但这在实际开发中是不可能的, 我们的类会不断迭代, 一旦类被修改了, 那旧对象反序列化就会报错. 所以在实际开发中, 我们都会显示指定一个 serialVersionUID, 值是多少无所谓, 只要不变就行.
    写个实例测试下:
    (1) User 类
    不显示指定 serialVersionUID.

    1. public class User implements Serializable {
    2. private String name;
    3. private Integer age;
    4. public String getName() {
    5. return name;
    6. }
    7. public void setName(String name) {
    8. this.name = name;
    9. }
    10. public Integer getAge() {
    11. return age;
    12. }
    13. public void setAge(Integer age) {
    14. this.age = age;
    15. }
    16. @Override
    17. public String toString() {
    18. return "User{" +
    19. " + name + '\\'' +
    20. ", age=" + age +
    21. '}';
    22. }
    23. }

    (2) 测试类
    先进行序列化, 再进行反序列化. ```java public class SerializableTest {

    private static void serialize(User user) throws Exception {

    1. ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(new File("D:\\\\111.txt")));
    2. oos.writeObject(user);
    3. oos.close();

    }

    private static User deserialize() throws Exception{

    1. ObjectInputStream ois = new ObjectInputStream(new FileInputStream(new File("D:\\\\111.txt")));
    2. return (User) ois.readObject();

    }

  1. public static void main(String\[\] args) throws Exception {
  2. User user = new User();
  3. user.setName("tyshawn");
  4. user.setAge(18);
  5. System.out.println("序列化前的结果: " + user);
  6. serialize(user);
  7. User dUser = deserialize();
  8. System.out.println("反序列化后的结果: "+ dUser);
  9. }

}

  1. (3) 结果<br />先注释掉反序列化代码, 执行序列化代码, 然后 User 类新增一个属性 sex
  2. ```java
  3. public class User implements Serializable {
  4. private String name;
  5. private Integer age;
  6. private String sex;
  7. public String getName() {
  8. return name;
  9. }
  10. public void setName(String name) {
  11. this.name = name;
  12. }
  13. public Integer getAge() {
  14. return age;
  15. }
  16. public void setAge(Integer age) {
  17. this.age = age;
  18. }
  19. public String getSex() {
  20. return sex;
  21. }
  22. public void setSex(String sex) {
  23. this.sex = sex;
  24. }
  25. @Override
  26. public String toString() {
  27. return "User{" +
  28. " + name + '\\'' +
  29. ", age=" + age +
  30. ", sex='" + sex + '\\'' +
  31. '}';
  32. }
  33. }

再注释掉序列化代码执行反序列化代码, 最后结果如下:
序列化前的结果: User{name=’tyshawn’, age=18} Exception in thread “main” java.io.InvalidClassException: org.tyshawn.SerializeAndDeserialize.User; local class incompatible: stream classdesc serialVersionUID = 1035612825366363028, local class serialVersionUID = -1830850955895931978
报错结果为序列化与反序列化产生的 serialVersionUID 不一致.
接下来我们在上面 User 类的基础上显示指定一个 serialVersionUID

  1. private static final long serialVersionUID = 1L;

显示指定 serialVersionUID 后就解决了序列化与反序列化产生的 serialVersionUID 不一致的问题.

Java 序列化的其他特性

先说结论, 被 transient 关键字修饰的属性不会被序列化, static 属性也不会被序列化.
我们来测试下这个结论:
(1) User 类

  1. public class User implements Serializable {
  2. private static final long serialVersionUID = 1L;
  3. private String name;
  4. private Integer age;
  5. private transient String sex;
  6. private static String signature = "你眼中的世界就是你自己的样子";

https://mp.weixin.qq.com/s/j0gYgWT5tWwU8G7M2GmX2g