stream classdesc serialVersionUID = 1, local class serialVersionUID = 2

原创转载请注明出处:http://agilestyle.iteye.com/blog/2301301

先看一个简单的例子

package org.fool.test;

import java.io.Serializable;

public class User implements Serializable {

	/**
	 * 
	 */
	private static final long serialVersionUID = 1L;

	private long id;
	private String username;
	private String password;

	public User() {
	}

	public User(long id, String username, String password) {
		this.id = id;
		this.username = username;
		this.password = password;
	}

	@Override
	public String toString() {
		return "User [id=" + id + ", username=" + username + ", password=" + password + "]";
	}

	public long getId() {
		return id;
	}

	public void setId(long id) {
		this.id = id;
	}

	public String getUsername() {
		return username;
	}

	public void setUsername(String username) {
		this.username = username;
	}

	public String getPassword() {
		return password;
	}

	public void setPassword(String password) {
		this.password = password;
	}
}

 这里先创建一个User对象,序列化UID为1L

package org.fool.test;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

public class SerializableTest {
	public static void main(String[] args) throws Exception {
		ObjectOutputStream oos = new ObjectOutputStream(new BufferedOutputStream(new FileOutputStream(new File("serial.txt"))));
		oos.writeObject(new User(1L, "hello", "world"));
		oos.close();
		
		ObjectInputStream ois = new ObjectInputStream(new BufferedInputStream(new FileInputStream(new File("serial.txt"))));
		User user = (User) ois.readObject();
		System.out.println(user);
		ois.close();
	}
}

 接着使用IO进行读写,查看输出

User [id=1, username=hello, password=world]


本地工程下面生成了serial.txt文件,然后将User对象的序列化UID改为2L

public class User implements Serializable {

	/**
	 * 
	 */
	private static final long serialVersionUID = 2L;

	private long id;
	private String username;
	private String password;
......

 同时注释掉原先的IO写操作,只进行读操作

public class SerializableTest {
	public static void main(String[] args) throws Exception {
		//ObjectOutputStream oos = new ObjectOutputStream(new BufferedOutputStream(new FileOutputStream(new File("serial.txt"))));
		//oos.writeObject(new User(1L, "hello", "world"));
		//oos.close();
		
		ObjectInputStream ois = new ObjectInputStream(new BufferedInputStream(new FileInputStream(new File("serial.txt"))));
		User user = (User) ois.readObject();
		System.out.println(user);
		ois.close();
	}
}

 查看输出

Exception in thread "main" java.io.InvalidClassException: org.fool.test.User; local class incompatible: stream classdesc serialVersionUID = 1, local class serialVersionUID = 2
	at java.io.ObjectStreamClass.initNonProxy(ObjectStreamClass.java:616)
	at java.io.ObjectInputStream.readNonProxyDesc(ObjectInputStream.java:1623)
	at java.io.ObjectInputStream.readClassDesc(ObjectInputStream.java:1518)
	at java.io.ObjectInputStream.readOrdinaryObject(ObjectInputStream.java:1774)
	at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1351)
	at java.io.ObjectInputStream.readObject(ObjectInputStream.java:371)
	at org.fool.test.SerializableTest.main(SerializableTest.java:18)

Java通过一个名为UID(stream unique identifier)来控制,这个UID是隐式的,它通过类名,方法名等诸多因素经过计算而得,理论上是一一映射的关系,也就是唯一的。如果UID不一样的话,就无法实现反序列化了,并且将会得到InvalidClassException。这就是通常项目中版本升级会遇到的问题,很有可能缓存或者数据库的数据是以1L的版本写入,之后由于项目版本升级,UID变为了2L,在2L的基础上进行读操作,就会抛出这个异常,也就是序列化兼容性问题,解决的办法就是将UID版本统一,并且将原先缓存或数据库的数据清空,同时以当前最新的UID版本重新写入。

猜你喜欢

转载自agilestyle.iteye.com/blog/2301301
今日推荐