[Java开发之路](9)对象序列化与反序列化
1. 对象序列化
当然,我们也能够通过将信息写入文件或者数据库,可是假设能将一个对象声明为是"持久性"的,并为我们处理掉全部的细节,这将会显得十分方便。
这意味着序列化机制能自己主动弥补不同操作系统之间的差异。
一是Java的远程方法调用(RMI),它使存活于其它计算机上的对象使用起来就像存活于本机上一样。
当向远程对象发送消息时,须要通过对象序列化来传输參数和返回值。二是Java Beans。使用一个bean时。普通情况下是在设计阶段对它的状态信息进行配置。这样的状态信息必须保存下来,并在程序启动时进行后期恢复(这样的详细工作就是由对象序列化完毕的)。
这时,仅仅需调用writeObject()就可以将对象序列化,并将其发送给 OutputStream(对象序列化是基于字节的,因要使用InputStream和OutputStream继承层次结构)。
要反向进行该过程(将一个序列化还原为一个对象),须要将一个InputStream封装在ObjectInputStream内,然后调用readObject()。和往常一样。我们最后获得的是一个引用,它指向一个向上转型的Object,所以必须向下转型才干直接设置它们。
package com.qunar.bean;import java.io.Serializable;/*** 学生实体类* @author sjf0115**/public class Student implements Serializable{/****/private static final long serialVersionUID = 1L;// 姓名private String name;// 学号private String ID;// 年龄private int age;// 学校private String school;/*** @param name* @param iD* @param age* @param school*/public Student(String name, String id, int age, String school) {super();this.name = name;ID = id;this.age = age;this.school = school;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getID() {return ID;}public void setID(String iD) {ID = iD;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String getSchool() {return school;}public void setSchool(String school) {this.school = school;}@Overridepublic String toString() {return "姓名:" + name + " 学号:" + ID + " 年龄:" + age + " 学校:" + school;}}
package com.qunar.io;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.FileOutputStream;import java.io.IOException;import java.io.ObjectInputStream;import java.io.ObjectOutputStream;import com.qunar.bean.Student;public class SeriaCode {public static void main(String[] args) {// 对象序列化数据保存位置String path = "D:\\seria.dat";try {// 创建FileOutputStream对象FileOutputStream fileOutputStream = new FileOutputStream(path);// 创建ObjectOutputStream对象ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);// 序列化对象Student student = new Student("xiaosi","130346",25,"西安电子科技大学");// 进行对象序列化 Student对象要实现序列化接口objectOutputStream.writeObject(student);objectOutputStream.flush();objectOutputStream.close();// 创建FileInputStream对象FileInputStream fileInputStream = new FileInputStream(path);// 创建ObjectInputStream对象ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);// 反序列化为对象Student stu = (Student)objectInputStream.readObject();objectInputStream.close();System.out.println("Stu->"+stu);} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();}}}
package com.qunar.io.serial;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.IOException;import java.io.ObjectInputStream;public class SerialCode2 {public static void main(String[] args) {try {// 创建FileInputStream对象FileInputStream fileInputStream = new FileInputStream("seria.dat");// 创建ObjectInputStream对象ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);// 反序列化为对象Object object = objectInputStream.readObject();objectInputStream.close();System.out.println(object.getClass());} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();}}}
|
java.lang.ClassNotFoundException: com.qunar.io.Student
at java.net.URLClassLoader$1.run(Unknown Source)
at java.net.URLClassLoader$1.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at sun.misc.Launcher$AppClassLoader.loadClass(Unknown Source)
at java.lang.ClassLoader.loadClass(Unknown Source)
at java.lang.Class.forName0(Native Method)
at java.lang.Class.forName(Unknown Source)
at java.io.ObjectInputStream.resolveClass(Unknown Source)
at java.io.ObjectInputStream.readNonProxyDesc(Unknown Source)
at java.io.ObjectInputStream.readClassDesc(Unknown Source)
at java.io.ObjectInputStream.readOrdinaryObject(Unknown Source)
at java.io.ObjectInputStream.readObject0(Unknown Source)
at java.io.ObjectInputStream.readObject(Unknown Source)
at com.qunar.io.serial.SerialCode2.main(SerialCode2.java:17)
|
这两个方法会在序列化和反序列化还原过程中自己主动调用。以便运行一些特殊操作。
package com.qunar.io;import java.io.Externalizable;import java.io.IOException;import java.io.ObjectInput;import java.io.ObjectOutput;public class Fruit implements Externalizable{public Fruit(){System.out.println("Fruit constructor...");}@Overridepublic void writeExternal(ObjectOutput out) throws IOException {System.out.println("Fruit writeExternal...");}@Overridepublic void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {System.out.println("Fruit readExternal...");}}
package com.qunar.io;import java.io.Externalizable;import java.io.IOException;import java.io.ObjectInput;import java.io.ObjectOutput;public class Fruit2 implements Externalizable{Fruit2(){System.out.println("Fruit2 constuctor...");}@Overridepublic void writeExternal(ObjectOutput out) throws IOException {System.out.println("Fruit writeExternal...");}@Overridepublic void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {System.out.println("Fruit readExternal...");}}
package com.qunar.io;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.FileOutputStream;import java.io.IOException;import java.io.ObjectInputStream;import java.io.ObjectOutputStream;public class FruitSerialCode {public static void main(String[] args) {try {// 创建FileOutputStream对象FileOutputStream fileOutputStream = new FileOutputStream("fruit.out");// 创建ObjectOutputStream对象ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);// 序列化对象Fruit fruit = new Fruit();Fruit2 fruit2 = new Fruit2();// 进行对象序列化 Fruit对象要实现序列化接口System.out.println("writeObject...");objectOutputStream.writeObject(fruit);objectOutputStream.writeObject(fruit2);objectOutputStream.flush();objectOutputStream.close();// 创建FileInputStream对象FileInputStream fileInputStream = new FileInputStream("fruit.out");// 创建ObjectInputStream对象ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);// 反序列化为对象System.out.println("readFruit...");fruit = (Fruit)objectInputStream.readObject();System.out.println("readFruit2...");fruit2 = (Fruit2)objectInputStream.readObject();objectInputStream.close();} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();}}}
|
Fruit constructor...
Fruit2 constuctor...
writeObject...
Fruit writeExternal...
Fruit writeExternal...
readFruit...
Fruit constructor...
Fruit readExternal...
readFruit2...
java.io.InvalidClassException: com.qunar.io.Fruit2; no valid constructor
at java.io.ObjectStreamClass$ExceptionInfo.newInvalidClassException(Unknown Source)
at java.io.ObjectStreamClass.checkDeserialize(Unknown Source)
at java.io.ObjectInputStream.readOrdinaryObject(Unknown Source)
at java.io.ObjectInputStream.readObject0(Unknown Source)
at java.io.ObjectInputStream.readObject(Unknown Source)
at com.qunar.io.FruitSerialCode.main(FruitSerialCode.java:36)
|
上例中没有反序列化后Fruit2对象。而且导致了一个异常。主要是Fruit的构造函数是public的,而Fruit2的构造函数却不是,这样就会在反序列时抛出异常。
对于一个Serializable对象。对象全然以它存储的二进制位为基础,而不用调用构造函数。而对于一个Externalizable对象,全部普通的构造函数都会被调用(包含在字段定义时的初始化)。然后调用readExternal()。
|
全部默认的构造函数都会被调用。才干使Externalizable对象产生正确的行为。 |
package com.qunar.io;import java.io.Externalizable;import java.io.IOException;import java.io.ObjectInput;import java.io.ObjectOutput;public class Fruit implements Externalizable{private String name;private int num;// 必须有默认构造函数 反序列时使用public Fruit(){System.out.println("Fruit default constructor...");}/*** @param name* @param num*/public Fruit(String name, int num) {System.out.println("Fruit constructor...");this.name = name;this.num = num;}@Overridepublic void writeExternal(ObjectOutput out) throws IOException {System.out.println("Fruit writeExternal...");// 必须做例如以下操作out.writeObject(name);out.writeInt(num);}@Overridepublic void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {System.out.println("Fruit readExternal...");// 必须做例如以下操作name = (String)in.readObject();num = in.readInt();}@Overridepublic String toString() {return "name:" + name + " num:" + num;}}
package com.qunar.io;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.FileOutputStream;import java.io.IOException;import java.io.ObjectInputStream;import java.io.ObjectOutputStream;public class FruitSerialCode {public static void main(String[] args) {try {// 创建FileOutputStream对象FileOutputStream fileOutputStream = new FileOutputStream("fruit.out");// 创建ObjectOutputStream对象ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);// 序列化对象Fruit fruit = new Fruit("苹果",20);// 进行对象序列化 Fruit对象要实现序列化接口System.out.println("writeObject...");objectOutputStream.writeObject(fruit);objectOutputStream.flush();objectOutputStream.close();// 创建FileInputStream对象FileInputStream fileInputStream = new FileInputStream("fruit.out");// 创建ObjectInputStream对象ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);// 反序列化为对象System.out.println("readFruit...");fruit = (Fruit)objectInputStream.readObject();System.out.println("Fruit->[" + fruit + "]");objectInputStream.close();} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();}}}
|
Fruit constructor...
writeObject...
Fruit writeExternal...
readFruit...
Fruit default constructor...
Fruit readExternal...
Fruit->[name:苹果 num:20]
|
|
Fruit constructor...
writeObject...
Fruit writeExternal...
readFruit...
Fruit default constructor...
Fruit readExternal...
Fruit->[name:null num:0]
|
为了进行控制,使用transientkeyword关闭序列化操作,它的意思"不用麻烦你序列化或者反序列化数据,我自己会处理的"。
package com.qunar.io;import java.io.Serializable;import java.util.Date;public class Login implements Serializable{/****/private static final long serialVersionUID = 1L;private Date date = new Date();private String userName;// 防止被序列化transientprivate transient String password;/*** @param date* @param userName* @param password*/public Login(String userName, String password) {super();this.userName = userName;this.password = password;}@Overridepublic String toString() {return "Date:" + date + " UserName:" + userName + " Password:" + password;}}
package com.qunar.io;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.FileOutputStream;import java.io.IOException;import java.io.ObjectInputStream;import java.io.ObjectOutputStream;public class LoginSerialCode {public static void main(String[] args) {try {// 创建FileOutputStream对象FileOutputStream fileOutputStream = new FileOutputStream("login.out");// 创建ObjectOutputStream对象ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);// 序列化对象Login login = new Login("xiaosi", "123");// 进行对象序列化 Fruit对象要实现序列化接口System.out.println("writeObject...");objectOutputStream.writeObject(login);objectOutputStream.flush();objectOutputStream.close();// 创建FileInputStream对象FileInputStream fileInputStream = new FileInputStream("login.out");// 创建ObjectInputStream对象ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);// 反序列化为对象System.out.println("readFruit...");login = (Login)objectInputStream.readObject();System.out.println("LoginInfo->[" + login + "]");objectInputStream.close();} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();}}}
|
writeObject...
readFruit...
LoginInfo->[Date:Thu Dec 31 00:16:13 CST 2015 UserName:xiaosi Password:null]
|
同一时候我们发现date字段被存储在磁盘而且从磁盘上恢复出来,而不是又一次生成。
这样一旦进行序列化和反序列化,就会自己主动的分别调用这两个方法,来取代默认的序列化机制。
package com.qunar.io;import java.io.IOException;import java.io.ObjectInputStream;import java.io.ObjectOutputStream;import java.io.Serializable;import java.util.Date;public class Login implements Serializable{/****/private static final long serialVersionUID = 1L;private Date date = new Date();private String userName;// 防止被序列化transientprivate transient String password;/*** @param date* @param userName* @param password*/public Login(String userName, String password) {super();this.userName = userName;this.password = password;}// 必须有private void writeObject(ObjectOutputStream stream) throws IOException{// 默认的序列化stream.defaultWriteObject();// 手动完毕序列化stream.writeObject(password);}// 必须有private void readObject(ObjectInputStream stream) throws ClassNotFoundException, IOException{// 默认的反序列化stream.defaultReadObject();// 手动完毕反序列化password = (String)stream.readObject();}@Overridepublic String toString() {return "Date:" + date + " UserName:" + userName + " Password:" + password;}}
package com.qunar.io;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.FileOutputStream;import java.io.IOException;import java.io.ObjectInputStream;import java.io.ObjectOutputStream;public class LoginSerialCode {public static void main(String[] args) {try {// 创建FileOutputStream对象FileOutputStream fileOutputStream = new FileOutputStream("login.out");// 创建ObjectOutputStream对象ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);// 序列化对象Login login = new Login("xiaosi", "123");// 进行对象序列化 Fruit对象要实现序列化接口System.out.println("writeObject...");objectOutputStream.writeObject(login);objectOutputStream.flush();objectOutputStream.close();// 创建FileInputStream对象FileInputStream fileInputStream = new FileInputStream("login.out");// 创建ObjectInputStream对象ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);// 反序列化为对象System.out.println("readFruit...");login = (Login)objectInputStream.readObject();System.out.println("LoginInfo->[" + login + "]");objectInputStream.close();} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();}}}
|
writeObject...
readFruit...
LoginInfo->[Date:Fri Jan 01 15:57:53 CST 2016 UserName:xiaosi Password:123]
|
在这个样例中。password字段是transient字段,用来证明非transient字段是由defaultWriteObject()方法保存,而transient字段是必须在程序中明白保存和恢复。
package com.qunar.io;import java.io.Serializable;public class House implements Serializable{/****/private static final long serialVersionUID = 1L;private String name;private String location;/*** @param name* @param location*/public House(String name, String location) {super();this.name = name;this.location = location;}}
package com.qunar.io;import java.io.Serializable;public class Animal implements Serializable{/****/private static final long serialVersionUID = 1L;private String name;private House house;/*** 构造函数* @param name* @param house*/public Animal(String name, House house) {super();this.name = name;this.house = house;}@Overridepublic String toString() {return name + " " + house;}}
package com.qunar.io;import java.io.ByteArrayInputStream;import java.io.ByteArrayOutputStream;import java.io.FileNotFoundException;import java.io.IOException;import java.io.ObjectInputStream;import java.io.ObjectOutputStream;import java.util.ArrayList;import java.util.List;public class AnimalSerialCode {@SuppressWarnings("unchecked")public static void main(String[] args) {House house = new House("水立方","北京海淀区");List<Animal> animals = new ArrayList<Animal>();animals.add(new Animal("狗", house));animals.add(new Animal("鸡", house));animals.add(new Animal("羊", house));System.out.println("Animals->" + animals);try {ByteArrayOutputStream buf = new ByteArrayOutputStream();ObjectOutputStream objectOutputStream = new ObjectOutputStream(buf);// 序列化objectOutputStream.writeObject(animals);objectOutputStream.writeObject(animals);ByteArrayOutputStream buf2 = new ByteArrayOutputStream();ObjectOutputStream objectOutputStream2 = new ObjectOutputStream(buf2);// 序列化objectOutputStream2.writeObject(animals);ObjectInputStream objectInputStream = new ObjectInputStream(new ByteArrayInputStream(buf.toByteArray()));List<Animal> ani1 = (List<Animal>)objectInputStream.readObject();List<Animal> ani2 = (List<Animal>)objectInputStream.readObject();ObjectInputStream objectInputStream2 = new ObjectInputStream(new ByteArrayInputStream(buf2.toByteArray()));List<Animal> ani3 = (List<Animal>)objectInputStream2.readObject();System.out.println("Animals1->"+ani1);System.out.println("Animals2->"+ani2);System.out.println("Animals3->"+ani3);} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {// TODO Auto-generated catch blocke.printStackTrace();}}}
|
Animals->[狗 com.qunar.io.House@1b15692, 鸡 com.qunar.io.House@1b15692, 羊 com.qunar.io.House@1b15692]
Animals1->[狗 com.qunar.io.House@1aaf0b3, 鸡 com.qunar.io.House@1aaf0b3, 羊 com.qunar.io.House@1aaf0b3]
Animals2->[狗 com.qunar.io.House@1aaf0b3, 鸡 com.qunar.io.House@1aaf0b3, 羊 com.qunar.io.House@1aaf0b3]
Animals3->[狗 com.qunar.io.House@1a082e2, 鸡 com.qunar.io.House@1a082e2, 羊 com.qunar.io.House@1a082e2]
|
在这个样例中,Animal对象包括House类型字段。我们创建Animals列表并将其两次序列化,分别送至不同的流。当期被反序列化还原被打印时。我们能够看到:每次执行时对象将会处在不同的内存地址。
[Java开发之路](9)对象序列化与反序列化的更多相关文章
- java 对象序列化与反序列化
Java序列化与反序列化是什么? 为什么需要序列化与反序列化? 如何实现Java序列化与反序列化? 本文围绕这些问题进行了探讨. 1.Java序列化与反序列化 Java序列化是指把Java对象转换为 ...
- Java Io 对象序列化和反序列化
Java 支持将任何对象进行序列化操作,序列化后的对象文件便可通过流进行网络传输. 1. 对象序列化就是将对象转换成字节序列,反之叫对象的反序列化 2. 序列化流ObjectOut ...
- Java对象序列化与反序列化一 JSON
Java对象序列化与反序列化一 JSON 1. 依赖库 jackson-all-1.6.1.jar 2. 代码 public class Student { private String nam ...
- Java之对象序列化和反序列化
一.对象序列化和反序列化存在的意义: 当你创建对象,只要你需要,他就一直存在,但当程序结束,对象就会消失,但是存在某种情况,如何让程序在不允许的状态,仍然保持该对象的信息.并在下次程序运行的时候使用该 ...
- Java 序列化 对象序列化和反序列化
Java 序列化 对象序列化和反序列化 @author ixenos 对象序列化是什么 1.对象序列化就是把一个对象的状态转化成一个字节流. 我们可以把这样的字节流存储为一个文件,作为对这个对象的复制 ...
- Java IO详解(六)------序列化与反序列化(对象流)
File 类的介绍:http://www.cnblogs.com/ysocean/p/6851878.html Java IO 流的分类介绍:http://www.cnblogs.com/ysocea ...
- Java对象序列化和反序列化的工具方法
import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import ja ...
- Java对象序列化与反序列化
对象序列化的目标是将对象保存在磁盘中或者在网络中进行传输.实现的机制是允许将对象转为与平台无关的二进制流. java中对象的序列化机制是将允许对象转为字节序列.这些字节序列可以使Java对象脱离程序存 ...
- Java基础(二十九)Java IO(6)对象序列化(Object Serialization)
参考之前整理过的Java序列化与反序列化:https://www.cnblogs.com/BigJunOba/p/9127414.html 使用对象输入输出流可以实现对象序列化与反序列化,可以直接存取 ...
- C#对象序列化与反序列化zz
C#对象序列化与反序列化(转载自:http://www.cnblogs.com/LiZhiW/p/3622365.html) 1. 对象序列化的介绍........................ ...
随机推荐
- thinkphp5项目--个人博客(三)
thinkphp5项目--个人博客(三) 项目地址 fry404006308/personalBlog: personalBloghttps://github.com/fry404006308/per ...
- 福建省赛--Problem E The Longest Straight(标记+二分)
Problem E The Longest Straight Accept: 71 Submit: 293 Time Limit: 1000 mSec Memory Limit : 327 ...
- BZOJ 1379 模拟退火
模拟退火的第一题~ //By SiriusRen #include <cmath> #include <cstdio> #include <algorithm> u ...
- 在Ubuntu下使用命令删除目录
在Ubuntu命令行中用命令删除目录,现在在Linux系统中删除目录大致会用两个,rm和rmdir,rm命令删除目录很简单,不过很多人还是比较习惯用rmdir命令,如果操作的目录非空时就有点麻烦.这时 ...
- Maven项目:Plugin execution not covered by lifecycle configuration 解决方案
这个是eclipse中配置文件pom.xml报的错.具体错误信息: Plugin execution not covered by lifecycle configuration: org.apach ...
- ORM原理
原理: 1.实现JavaBean的属性到数据库表的字段的映射: --通过配置文件将JavaBean的属性与数据库表的字段的关联起来 2.映射关系: 一对多,多对一等 持久层(Pers ...
- centos7 安装 Python netsnmp模块
由于模块的安装过程中有点艰难(其实挺简单,只是参照网上的教程很多都装不成功,花了很多时间...),所以记下来备忘. 先装一下python-devel 模块,执行命令 yum install pytho ...
- webStrom的破解以及汉化
破解方法: 把JetbrainsCrack-3.1-release-enc.jar包放到bin目录下,然后把webstorm64.exe.vmoptions文件用文本打开, 在最后面加上一句-java ...
- FastDFS学习总结(1)--FastDFS安装和部署
FastDFS是一个开源的,高性能的的分布式文件系统,他主要的功能包括:文件存储,同步和访问,设计基于高可用和负载均衡,FastDFS非常适用于基于文件服务的站点,例如图片分享和视频分享网站 Fast ...
- Visual Studio2008 和2010 执行程序出现的黑框马上消失解决方法
1 在程序最后加 system("PAUSE"); 要注意包括头文件#include"stdlib.h" //system须要调用这个 ...