`

java序列化与反序列化

 
阅读更多

java序列化与反序列化

java序列化与反序列化的目的是:存储和重新恢复建立Java对象。序列化可以使Java对象更好地传输和存储。反序列化就是把序列化的Java对象重新恢复为Java对象的格式。

(以上为个人简要的理解,标准的解释可参见官网文档:http://docs.oracle.com/javase/8/docs/platform/serialization/spec/serial-arch.html)

 

package com.ghost.t20140502;

import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

public class TestSerial{

	public static void main(String[] args){
		
		try {
			//序列化SerialObject对象,并把它保存到工作空间的SerialObject.txt文件中
			FileOutputStream fos = new FileOutputStream("SerialObject.txt");
			ObjectOutputStream oos = new ObjectOutputStream(fos);
			SerialObject so = new SerialObject(); 
			oos.writeObject(so); 
			oos.flush(); 
			oos.close();
			//读取SerialObject.txt中的信息,并把它反序列化为Java对象
			FileInputStream fis = new FileInputStream("SerialObject.txt");   
			ObjectInputStream oin = new ObjectInputStream(fis);  
			SerialObject ts = (SerialObject) oin.readObject(); 
			oin.close();
			//读取反序列化后对象中的值
			System.out.println("version:"+ts.version); 
			System.out.println("version:"+ts.getName());
		} catch (Exception e) {
			e.printStackTrace();
		}finally{
			
		}
	}
}
class SerialObject implements Serializable{// 必须实现Serializable接口;Serializable这个是一个标识接口,让Java虚拟机运行时知道该类的对象可以序列化
	private static final long serialVersionUID = 5843403696945102665L;
	public byte version = 1;
	private name = "java";
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
}

 

private static final long serialVersionUID = 5843403696945102665L;

Java的序列化机制是通过在运行时判断类的serialVersionUID来验证版本一致性的。在进行反序列化时,JVM会把传来 的字节流中的serialVersionUID与本地相应实体(类)的serialVersionUID进行比较,如果相同就认为是一致的,可以进行反序 列化,否则就会出现序列化版本不一致的异常。serialVersionUID这个字段就相当于Java类的指纹。 Java类的某个对象序列化后的字节,称为a,a可以存储于内存、硬盘,可以通过网络传输。当我们需要重新获得a的信息时,则需要反序列化对应到本地运行中的Java类,该类称为A。当初a可能并不是根据A来序列化的,也就是a对象对应的类并不一定是A。JVM就是根据serialVersionUID的值来判定a所对应的类。

 

当a中的信息(字段)多于类A中的字段时,没关系,因为反序列化是以A类来看待a对象的,a中多的信息只是无法在A上提取出来而已,也就是用A来操作a获得不到a中有而A上没有的字段。而A上有而a上没有的字段,也没关系,因为当a反序列化为A的对象时,会对所有属性赋值(没有对应的则赋初值)。Java生成的每个对象的字段都至少有默认值。

serialVersionUID的值是如何算出来的呢?它主要依靠的信息是Java源代码编译后的Java字节码,根据Java字节码中的信息来计算的。像Java源码中的注释、空格之类肯定是不会影响serialVersionUID的值的。

如果不加serialVersionUID这个会怎样?可以如此测试,先生成SerialObject的一个对象的序列化文件s。然后我们改变SerialObject类,比如增加一个字段public String namea = "java"; 然后再把s反序列化为SerialObject对象,此时必然报错:

java.io.InvalidClassException: com.ghost.t20140502.SerialObject; local class incompatible: stream classdesc serialVersionUID = 5843403696945102665, local class serialVersionUID = -5027101552399591984

 

如果只是改变类中的初始值,比如public byte version = 1;改为public byte version = 2;  那么是不会影响的,当然从s中读出来的还是1。

由此我们知道serialVersionUID的值必然存在序列化后的文件中。当没有对类加serialVersionUID字段时,序列化时,Java会根据类信息计算一个存到序列化文件中。

 

 可以用能打开十六进制的文本编译器UltraEdit、EditPlus等打开SerialObject.txt这个文件:


可以了解看看、学习研究。
 

  • 大小: 7.9 KB
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics