深圳优定软件网站建设,江门做网站公司开网络公司,全国公共资源交易中心招标网,网站域名一年多少钱我们知道在对数据进行传输时#xff0c;需要将其进行序列化#xff0c;在Java中实现序列化的方式也很简单#xff0c;可以直接通过实现Serializable接口。但是我们经常也会看到下面接这一行代码#xff0c;private static final Long serialVersionUID 1L#xff1b;这段代…我们知道在对数据进行传输时需要将其进行序列化在Java中实现序列化的方式也很简单可以直接通过实现Serializable接口。但是我们经常也会看到下面接这一行代码private static final Long serialVersionUID 1L这段代码到底有什么用呢 为什么有些代码写了它有些代码没写一、案例代码1首先我们看这一段代码public class Person implements Serializable {private String name;private Integer age;public Person(){}public Person(String name, Integer age){this.name name;this.age age;}public String getName() {return name;}public void setName(String name) {this.name name;}public Integer getAge() {return age;}public void setAge(Integer age) {this.age age;}Overridepublic String toString() {return Person{ name name \ , age age };}
}public class IOTest {public static void main(String[] args) throws IOException, ClassNotFoundException {out();//in();}public static void out() throws IOException {ObjectOutputStream oos new ObjectOutputStream(new FileOutputStream(C:\\Users\\Lenovo\\Desktop\\java练习\\src\\main\\resources\\Person.txt));Person person new Person(张三,18);oos.writeObject(person);oos.close();}public static void in() throws IOException, ClassNotFoundException {ObjectInputStream ois new ObjectInputStream(new FileInputStream(C:\\Users\\Lenovo\\Desktop\\java练习\\src\\main\\resources\\Person.txt));Object object ois.readObject();System.out.println(object);}
}执行该段代码运行结果然后将out();注释掉, 执行in(); 如果这个时候我修改了Person类增加sex成员变量public class Person implements Serializable {private String name;private Integer age;private char sex;public Person(){}public Person(String name, Integer age, char sex) {this.name name;this.age age;this.sex sex;}public char getSex() {return sex;}public void setSex(char sex) {this.sex sex;}public Person(String name, Integer age){this.name name;this.age age;}public String getName() {return name;}public void setName(String name) {this.name name;}public Integer getAge() {return age;}public void setAge(Integer age) {this.age age;}Overridepublic String toString() {return Person{ name name \ , age age };}
}我在执行IOTest类中的in()方法(注释掉out();方法), 发现会报错这是为什么呢?为什么只是在Person类中增加了一个char sex成员变量, 执行in()方法就会报错.二、案例代码2如果在以上Person类中添加private static final long serialVersionUID 1L;按照案例代码1中的执行顺序运行一遍代码先将Person.txt文件中的内容全部删除这是为什么呢三、案例代码1和案例代码2分析这里先搞清楚java中的序列化和反序列化代表什么意思序列化把Java对象转换为字节序列的过程方序列化把字节序列恢复为Java对象的过程。这里看一下案例代码1执行所产生的的错误invalidClassException然后我们再看下面一行serialVersionUID有两个不同的值仔细分析发现这两个不同的值才是产生异常的原因。在案例1代码中我们在Person类中没有设置serialVersionUID值。我们首先执行的是out方法作用就是将person对象写入序列化到Person.txt文件中而在序列化的过程中Person类也会产生一个serialVersionUID值如果Person类中没有定义serialVersionUID值则系统会自动生成一个值就是第一个5428155211170925772会存入到Person.txt文件中然后我们在Person类中增加了一个sex成员变量注释掉out()方法执行in()方法相当于一个反序列化的过程这个时候会报错是因为在Person类由于被增加了一个sex成员变量导致系统生成的serialVersionUID改变了变成了-1544469845932994986而我们存入到Person.txt文件中的person对象的serialVersionUID却是5428155211170925772所以系统就判定这两个Person类不是同一个就会导致invalidClassException异常。案例2代码中我们规定了Person类中private static final long serialVersionUID 1L的值这就保证了无论Person类再怎么变serialVersionUID都是一个固定的值不再是系统生成的这就能够避免Person类增加或者减少成员变量都不会导致serialVersionUID改变。这是java源码中的一段话如果可序列化类没有显式声明serialVersionUID那么序列化运行时将根据类的各个方面计算该类的默认serialversion UID值如Java对象序列化规范中所述。此规范将枚举类型的seriaVersionUID定义为0L。但是。强烈建议除枚举类型之外的所有可序列化类都显式声明serialVersionUID值因为默认的seriaverslon UID计算对类细节非常敏感,这些细节可能因编译器实现而有所不同。因此在反序列化过程中可能会导致意外的InvalidClassException。因此。为了保证不同java编译器实现中的serialVerslonUID值一致。可序列化类必须声明显式serialVersionUID值。此外。强烈建议显式serialVersionUID声明尽可能使用私有修饰符。因为此类声明仅适用于立即声明的类—serialVersionUID字段作为继承成员没有用处。数组类不能声明显式的serialversionUID因此它们总是具有默认的计算值但数组类不需要匹配serialVersion UID值。