序列化和反序列化

什么是java的序列化和反序列化

java序列化就是把java对象转换为字节序列的过程。对于transient和static修饰的属性不会被序列化。原因:transient的作用就是这样,static是因为序列化作用的是对象,static修饰后是属于类的,所以没用。

java反序列化就是把字节序列转换为java对象的过程。

为什么要序列化

很多应用中,需要对对象进行序列化,让它们离开内存,放入到硬盘中去,以便长期保存。比如session,当有10万用户并发访问,可能出现10万个session对象,内存可能吃不消,web容器会把一些session序列化到硬盘中,

需要用的时候,再还原回来。

序列化需要实现java.io.Serializable接口,序列化的时候有一个serialVersionUID参数,java序列化机制是通过运行时判断类的serialVersionUID来验证版本一致性。在进行反序列化,Java虚拟机会把传过来的字节流中的serialVersionUID和本地相应实体类的serialVersionUID进行比较,如果相同就认为是一致的实体类,可以进行反序列化,否则Java虚拟机会拒绝对这个实体类进行反序列化并抛出异常。serialVersionUID有两种生成方式:

1、默认的1L

2、根据类名、接口名、成员方法以及属性等来生成一个64位的Hash字段

 如果实现java.io.Serializable接口的实体类没有显式定义一个名为serialVersionUID、类型为long的变量时,java的序列化机制会根据.class文件自动生成一个serialVersionUID,只要.class文件不变,那么变异多少次都是一样的。

 对象如果要序列化必须实现Serializable接口,如果该类中引用了别的实例变量,引用的类型也必须要实现该接口,如果没有实现是会出错的(如果值是null不会报错)。

手动指定序列化过程

如果进行序列化、反序列化时,虚拟机会首先试图调用待序列化对象里的writeObject和readObject方法,进行自定义的序列化和反序列化。如果在对象中没有定义,那么会默认调用的是ObjectOutputStream的defaultWriteObject和ObjectInputStream的defaultReadObject方法。

public static void main(String[] args) throws Exception
	{
	    File file = new File("/home/tp/s.txt");
	    OutputStream os = new FileOutputStream(file);  
	    ObjectOutputStream oos = new ObjectOutputStream(os);
	    oos.writeObject(new SerializableObject("str0", "str1"));
	    oos.close();
	        
	    InputStream is = new FileInputStream(file);
	    ObjectInputStream ois = new ObjectInputStream(is);
	    SerializableObject so = (SerializableObject)ois.readObject();
	    System.out.println("str0 = " + so.getStr0());
	    System.out.println("str1 = " + so.getStr1());
	    ois.close();
	} 
package algorithm;

import java.io.Serializable;

public class SerializableObject implements Serializable{

	/**
	 * 
	 */
	private static final long serialVersionUID = 1L;
	private String str0;
    private transient String str1;
    private static String str2 = "abc";
    
    public SerializableObject(String str0, String str1)
    {
        this.str0 = str0;
        this.str1 = str1;
    }
    
    public String getStr0()
    {
        return str0;
    }

    public String getStr1()
    {
        return str1;
    }
    
    private void writeObject(java.io.ObjectOutputStream s) throws Exception
    {
        System.out.println("我想自己控制序列化的过程");
        s.defaultWriteObject();
        s.writeInt(str1.length());
        for (int i = 0; i < str1.length(); i++)
            s.writeChar(str1.charAt(i));
    }
    
    private void readObject(java.io.ObjectInputStream s) throws Exception
    {
        System.out.println("我想自己控制反序列化的过程");
        s.defaultReadObject();
        int length = s.readInt();
        char[] cs = new char[length];
        for (int i = 0; i < length; i++)
            cs[i] = s.readChar();
        str1 = new String(cs, 0, length);
    }
}

在对象中写writeObject和readObject的通常用法:先通过defaultWriteObject和defaultReadObject方法序列化和反序列化对象。在文件结尾追加需要额外序列化的内容/读取额外的内容。

为什么需要我们自己去序列化对象:因为使用默认的序列化并不安全,有的时候一些敏感的东西我们需要再次加密,这个时候就可以使用自己定义的writeObject和readObject方法来进行序列化和反序列化。具体做法,把需要再次加密的属性用transient字段修饰,然后自己去序列化它们。

使用java原生的序列化方式来表示对象的优缺点

1、纯粹的Java环境下这种方式可以很好地工作,因为它是Java自带的,也不需要第三方的Jar包的支持。

2、多语言环境下,使用Java序列化方式进行存储后,很难用其他语言还原出结果

3、占用的字节数比较大,而且序列化、反序列化效率也不高

对象表示有各种各样的方式,序列化只是其中的一种而已。也可以把对象xml化以及json化。

借鉴http://www.cnblogs.com/xrq730/p/4823684.html

原文地址:https://www.cnblogs.com/tp123/p/6387603.html