java 的序列化和反序列化

1.序列化是干什么的?

简单说就是为了保存在内存中的各种对象的状态,
也就是实例变量,不是方法,
并且可以把保存的对象状态再读出来。
虽然你可以用你自己的各种各样的方法来保存object states,
但是Java给你提供一种应该比你自己好的保存对象状态的机制,那就是序列化。

2.什么情况下需要序列化

  • 当你想把的内存中的对象状态保存到一个文件中或者数据库中时候;

  • 当你想用套接字在网络上传送对象的时候;

  • 当你想通过RMI传输对象的时候;

3.一个例子

序列化需要实现Serializable或者Externalizable 接口
下面的是实现Serializable接口,最后会提到Externalizable

import java.io.Serializable;  
  
public class Student implements Serializable  
{  
 private String name;  
 private char sex;  
 private int year;  
 private double gpa;  
  
 public Student(){  
 }  
 public Student(String name,char sex,int year,double gpa)  
 {  
  this.name = name;  
  this.sex = sex;  
  this.year = year;  
  this.gpa = gpa;  
 }  
 //....getxxx setxxx  方法
}  

import java.io.*;  

public class UseStudent  
{  
 public static void main(String[] args)  
 {  
  Student st = new Student("德玛西亚",'M',20,3.6);  
  File file = new File("student.txt");  
  try  
  {  
   file.createNewFile();  
  }  
  catch(IOException e)  
  {  
   e.printStackTrace();  
  }  
  try  
  {  
   //Student对象序列化过程  
   FileOutputStream fos = new FileOutputStream(file);  
   ObjectOutputStream oos = new ObjectOutputStream(fos);  
   oos.writeObject(st);  
   oos.flush();  
   oos.close();  
   fos.close();  
  
   //Student对象反序列化过程  
   FileInputStream fis = new FileInputStream(file);  
   ObjectInputStream ois = new ObjectInputStream(fis);  
   Student st1 = (Student) ois.readObject();  
   System.out.println("name = " + st1.getName());  
   System.out.println("sex = " + st1.getSex());  
   System.out.println("year = " + st1.getYear());  
   System.out.println("gpa = " + st1.getGpa());  
   ois.close();  
   fis.close();  
  }  
  catch(ClassNotFoundException e)  
  {  
   e.printStackTrace();  
  }  
  catch (IOException e)  
  {  
   e.printStackTrace();  
  }               
 }  
}  

4.使用transient

在一些特殊场景下,比如银行账户对象,出于保密考虑,
不希望对存款金额进行序列化。或者类的一些引用类型的成员是不可序列化的。
此时可以使用transient关键字修饰不想被或者不能被序列化的成员变量。
需要注意的是transient只能修饰属性(filed),不能修饰类或方法。
一个静态变量不管是否被transient修饰,均不能被序列化。

5.自定义序列化

transient提供了一种简洁的方式将被transient修饰的成员属性完全隔离在序列化机制之外。
这样子固然不错,但是Java还提供了一种自定义序列化机制让开发者更自由地控制如何序列化各个成员属性,
或者不序列化某些属性(与transient效果相同)。

5.1 writeObject和readObject

private void writeObject(ObjectOutputStream out)
private void readObject(ObjectInputStream in)

这两个方法和ObjectOutputStream及ObjectInputStream里对应的方法名称相同。
实际上,尽管这两个方法是private型的,但是仍然是在被序列化(或反序列化)阶段被外部类ObjectOutputStream(或ObjectInputStream)调用。
仅以序列化为例:
ObjectOutputStream在执行自己的writeObject方法前会先通过反射在要被序列化的对象的类中
查找有无自定义的writeObject方法,
如有的话,则会优先调用自定义的writeObject方法。
因为查找反射方法时使用的是getPrivateMethod,
所以自定以的writeObject方法的作用域要被设置为private。
通过自定义writeObject和readObject方法可以完全控制对象的序列化与反序列化。


/**
 * 序列化测试类
 */
public class SerialTest {

    public static void main(String[] args) {
        Person robin = new Person("robin", 29);
        String savePath = "object.txt";

        SerialTest test = new SerialTest();

        try {
            test.serialize(savePath, robin);
            Person person = (Person) test.deSerialize(savePath);
            
            System.out.println("Name:" + person.getName()
                             +" Age:" + person.getAge());
            
        } catch (IOException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } 
    }


/**
 * Person类,可序列化

class Person implements Serializable {

    private static final long serialVersionUID = -6412852654889352693L;

    /**
     * 姓名
     */
    private String name;
    /**
     * 年龄
     */
    private int age;

    public Person() {}

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    /*  略去get和set,请自行实现  */
    
    private void writeObject(ObjectOutputStream out) throws IOException{
        out.writeObject(name);
        out.writeInt(age + 1);
        System.out.println("my write");
    }
    
    private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException{
        this.name = "zhangsan";
        this.age = 30;
        System.out.println("my read");
    }
}

5.2 writeReplace和readResolve

writeReplace和readResolve是一种更彻底的序列化的机制,
它甚至可以将序列化的目标对象替换为其它的对象。
但是与writeObject和readObject不同的是,
这二者不是必须要一起使用的,而且尽量应分开使用。
若一起使用的话,只有writeReplace会生效。

6.使用Externalizable

一开始有提到过实现Externalizable接口也可以实现类的序列化。
使用这种方法,可以由开发者完全决定如何序列化和反序列化目标对象。
Externalizable接口提供了writeExternal和readExternal两个方法。
实际上这种方法和前面的自定义序列化方法很相似,
只是Externalizable强制自定义序列化。
在使用了Externalizable的类中仍可以使用writeReplace和readResolve方法。
使用Externalizable进行序列化较之使用Serializable性能略好,但是复杂度较高。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,530评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,403评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,120评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,770评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,758评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,649评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,021评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,675评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,931评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,751评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,410评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,004评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,969评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,042评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,493评论 2 343

推荐阅读更多精彩内容