理解serialVersionUID是什么?有什么用?如何生成?

138 阅读2分钟

1、什么是serialVersionUID?

serialVersionUID用作Serializable类中的版本控件。如果您没有显式声明serialVersionUID,JVM将根据您的Serializable类的各个方面自动为您执行此操作,如Java(TM)对象序列化规范中所述

2、SerialVersionUID示例

3、相同的serialVersionUID,在反序列化过程中没有问题

在Address.java中,将serialVersionUID更改为2L(它是1L),然后再次编译它。
“ InvalidClassException”会引发,因为您使用serialVersionUID“1L”编写了一个序列化类,但尝试使用更新的序列化类serialVersionUID“2L”将其检索回来。

4、什么时候应该更新你的serialVersionUID?

如果使用对可序列化类的某些不兼容的Java类型更改更新序列化类,则必须更新serialVersionUID。

5、默认的serialVersionUID有什么问题?

如果没有声明serialVersionUID,JVM将使用自己的算法生成默认的SerialVersionUID,您可以在此处检查算法。默认的serialVersionUID计算对类详细信息非常敏感,可能因不同的JVM实现而异,并且在反序列化过程中会导致意外的InvalidClassExceptions。

6、如何生成serialVersionUID

实现Serializable接口,然后在IDEA使用如下方法:
1、打开Preferences–>Editor–>Inspections,然后在右侧输入UID进行搜索(搜索方式比较快,也可以在java–>Serialization issues里找)。然后勾选Serializable class without 'serialVersionUID'后面的复选框。右侧Severity默认Warning即可。如下图:

以下为可运行示例代码:

package com.wenxiaowu.reflect;
 
import org.junit.Test;
 
import java.io.*;
 
/**
 * 理解serialVersionUID
 */
public class SerializableTest {
    @Test
    public void testWrite() {
        Address address = new Address();
        address.setStreet("wall street");
        try {
            FileOutputStream fout = new FileOutputStream("address.ser");
            ObjectOutputStream oos = new ObjectOutputStream(fout);
            oos.writeObject(address);
            oos.close();
            System.out.println("Done");
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
 
    @Test
    public void testRead() {
        Address address;
        try {
            FileInputStream fin = new FileInputStream("address.ser");
            ObjectInputStream ois = new ObjectInputStream(fin);
            address = (Address) ois.readObject();
            ois.close();
            System.out.println(address); // Street : wall street
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
 
    public static class Address implements Serializable {
        private static final long serialVersionUID = 1L;
        String street;
 
        public String getStreet() {
            return this.street;
        }
 
        public void setStreet(String street) {
            this.street = street;
        }
 
        @Override
        public String toString() {
            return new StringBuffer(" Street : ")
                    .append(this.street)
                    .toString();
        }
    }
 
}