转自:http://hi.baidu.com/futurekey/blog/item/fdc30213451512c2c2fd78af.html#lastcmt
什么是序列化:一种用来处理对象流的机制,就是将对象的内容进行流化,方便对流化后的对象进行读写操作,网络之间的传输以及持久化到数据库或文件系统中.
如何实现:实现Serializable接口
序列化:
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("test.dat"));
oos.writeObject(obj);
反序列化:
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("test.dat"));
Object obj = ois.readObject();
序列化特点:
1.如果某个类能够被序列化,其子类也可以被序列化
2.如果该类有父类,则分两种情况,如果该父类已经实现了可序列化接口,则其父类的相应字段及属性的处理和该类相同;如果该类的父类没有实现可序列化接口,则该类的父类所有的字段属性将不会序列化
3.声明为static和transient类型的成员数据不能被序列化,因为static代表类的状态,transient代表对象的临时数据
序列化相关的类和接口:
在java.io包中提供的涉及对象的序列化的类与接口有ObjectOutput接口、ObjectOutputStream类、ObjectInput接口、ObjectInputStream类
1.ObjectOutput接口:它继承DataOutput接口并且支持对象的序列化,其内的writeObject()方法实现存储一个对象 ObjectInput接口:它继承DataInput接口并且支持对象的序列化,其内的readObject()方法实现读取一个对象
2.ObjectOutputStream类:它继承OutputStream类并且实现ObjectOutput接口。利用该类来实现将对象存储(调用ObjectOutput接口中的writeObject()方法) ObjectInputStream类:它继承InputStream类并且实现ObjectInput接口。利用该类来实现读取一个对象(调用ObjectInput接口中的readObject()方法)
注:对于父类的处理,如果父类没有实现序列化接口,则其必须有默认的构造函数(即没有参数的构造函数)。否则编译的时候就会报错。在反序列化的时候,默认构造函数会被调用。但是若把父类标记为可以序列化,则在反序列化的时候,其默认构造函数不会被调用。这是因为Java 对序列化的对象进行反序列化的时候,直接从流里获取其对象数据来生成一个对象实例,而不是通过其构造函数来完成
有些属性不想序列化:
并不是对象的每个属性都需要序列化,如果你不想让某个属性序列化可以用transient来修饰,transient修饰的属性,在序列化的时候会自动跳过,不做保存
有些属性想序列化但又没有实现Serializable接口:
并不是所所有的类都能序列化,但有些时候又不得不对某个属性序列化,这样的话我们也必须把这些属性用transient修饰,但是不能写入怎么反序列化?用下面的方法可以实现:
在该类中定义如下方法:
private void readObject(ObjectInputStream in) throws IOException,
ClassNotFoundException {
in.defaultReadObject();
this.需要反序列化的属性=in.read相应的类型();
}
private void writeObject(ObjectOutputStream out) throws IOException,
ClassNotFoundException {
out.defaultWriteObject();
out.write相应的类型(this.需要序列化的属性);
}
注:这两个方法必须是private的,它会在(反)序列化的时候自动调用
通过上面可以知道,其实transient其实就是一个标识作用,在序列化的时候通过它来判断哪些需要序列化哪些不需要序列化