zoukankan      html  css  js  c++  java
  • 理解serialVersionUID是什么?有什么用?如何生成?

    如果您曾经实现过Serializable接口,则必须遇到此警告消息
    The serializable class xxx does not declare a static final serialVersionUID field of type long
    那么......什么是serialVersionUID?
    serialVersionUID用作Serializable类中的版本控件。如果您没有显式声明serialVersionUID,JVM将根据您的Serializable类的各个方面自动为您执行此操作,如Java(TM)对象序列化规范中所述
    1. SerialVersionUID示例
    上面的语句在开头有点难以理解(至少我做过),让我们开始一个例子来了解Serializable类如何使用SerialVersionUID来实现版本控制。
     
    1.1 Address.java
    serialVersionUID为1L的可序列化类。
    import java.io.Serializable;
     
    public class Address implements Serializable{
     
           private static final long serialVersionUID = 1L;
        
           String street;
           String country;
     
           public void setStreet(String street){
               this.street = street;
           }
     
           public void setCountry(String country){
               this.country = country;
           }
     
           public String getStreet(){
               return this.street;
           }
     
           public String getCountry(){
               return this.country;
           }
     
           @Override
           public String toString() {
               return new StringBuffer(" Street : ")
               .append(this.street)
               .append(" Country : ")
               .append(this.country).toString();
           }
    }
     
    1.2 WriteObject.java
    将Address对象写入/序列化为文件的简单类 - “c:\ address.ser”。
    import java.io.FileOutputStream;
    import java.io.ObjectOutputStream;
     
    public class WriteObject{
     
        public static void main (String args[]) {
         
           Address address = new Address();
           address.setStreet("wall street");
           address.setCountry("united states");
     
           try{
     
            FileOutputStream fout = new FileOutputStream("c:\address.ser");
            ObjectOutputStream oos = new ObjectOutputStream(fout);   
            oos.writeObject(address);
            oos.close();
            System.out.println("Done");
     
           }catch(Exception ex){
               ex.printStackTrace();
           } 
        }
    }
     
    1.3 ReadObject.java
    从文件中读取/反序列化Address对象的简单类 - “c:\ address.ser”。
    import java.io.FileInputStream;
    import java.io.ObjectInputStream;
     
    public class ReadObject{
     
       public static void main (String args[]) {
     
           Address address;
           
           try{
     
               FileInputStream fin = new FileInputStream("c:\address.ser");
               ObjectInputStream ois = new ObjectInputStream(fin);
               address = (Address) ois.readObject();
               ois.close();
     
               System.out.println(address);
               
           }catch(Exception ex){
               ex.printStackTrace(); 
           } 
       }
    }
    2.测试
    让我们做一些测试来演示serialVersionUID的使用。
    2.1相同的serialVersionUID
    相同的serialVersionUID,在反序列化过程中没有问题
    javac Address.java javac WriteObject.java javac ReadObject.java java WriteObject java ReadObject Street : wall street Country : united states
    复制
    2.2不同的serialVersionUID
    在Address.java中,将serialVersionUID更改为2L(它是1L),然后再次编译它。
    javac Address.java java ReadObject java.io.InvalidClassException: Address; local class incompatible: stream classdesc serialVersionUID = 1, local class serialVersionUID = 2 ... at ReadObject.main(ReadObject.java:14)
     
    “ InvalidClassException”会引发,因为您使用serialVersionUID“1L”编写了一个序列化类,但尝试使用更新的序列化类serialVersionUID“2L”将其检索回来。
    serialVersionUID必须在序列化和反序列化过程中匹配。
    什么时候应该更新你的serialVersionUID?
    如果使用对可序列化类的某些不兼容的Java类型更改更新序列化类,则必须更新serialVersionUID。
    有关可序列化类的兼容和不兼容Java类型更改的详细信息,请参阅Java对象序列化规范。
    3.默认的serialVersionUID有什么问题?
    如果没有声明serialVersionUID,JVM将使用自己的算法生成默认的SerialVersionUID,您可以在此处检查算法。
    默认的serialVersionUID计算对类详细信息非常敏感,可能因不同的JVM实现而异,并且在反序列化过程中会导致意外的InvalidClassExceptions。
    3.1客户端/服务器环境
    - 客户端在Windows中使用SUN的JVM。
    - 服务器在Linux中使用JRockit。
    客户端通过套接字向服务器发送带有默认生成的serialVersionUID(例如123L)的可序列化类,服务器可以在反序列化过程中生成不同的serialVersionUID(例如124L),并引发意外的InvalidClassExceptions。
    3.2文件/数据库环境
    - App#1在Windows中使用SUN的JVM。
    - App#2在Linux中使用JRockit。
    序列化允许保存到文件或数据库中。App#1默认生成serialVersionUID(例如123L)将可序列化类存储到数据库中,而App#2可能在反序列化过程中生成不同的serialVersionUID(例如124L),并引发意外的InvalidClassExceptions。
    您可以在此处查看JVM实现列表
    4.如何生成serialVersionUID
    您可以使用JDK“ serialver”或Eclipse IDE自动生成serialVersionUID,详见详细信息
    结论
    SUN强烈建议开发人员声明serialVersionUID以避免上面列出的不同JVM问题,但我建议您应该了解什么是序列化,serialVersionUID如何实现版本控制以及您的类需要使用序列化的原因。了解serialVersionUID概念优于任何推荐的盲目。
     
    本文参考:https://www.mkyong.com/java-best-practices/understand-the-serialversionuid/
    参考
     
  • 相关阅读:
    go语言编程之旅笔记5
    go语言编程之旅笔记4
    go语言编程之旅笔记3
    go语言编程之旅笔记1~2
    minikube使用记录
    Azure Sql : Could not find stored procedure 'sp_addlinkedserver'.
    Jenkins SVN WebDeploy远程服务器
    sqlserver使用cte实现某列按字符分隔成多行
    openstack安装记录
    C语言中的bzero函数
  • 原文地址:https://www.cnblogs.com/xuxinstyle/p/11394358.html
Copyright © 2011-2022 走看看