序列化的概念及使用场合

序列化就是把对象转化为字节序列并持久化保存,可以保存在内存中、磁盘文件系统,甚至通过网络传递,并能够在以后将这个字节序列完全恢复为原来的对象。

对象序列化的概念引入Java是为了支持两种主要特性:

  • 一是为了远程调用(Remote Method Invocation,RMI),它使存活于其他计算机上的对象使用起来就像是存活于本机上一样。当想远程对象发送消息时,需要通过对象序列化来传输参数和返回值。
  • 二是对于Java Beans来说,对象的序列化也是必须的。使用一个Bean时,一般情况下是在设计阶段对它的状态信息进行配置,而这种状态信息必须保存下来,并在程序启动时进行后期恢复。

序列化操作步骤:

  1. 要使得一个对象可以被序列化,首先要实现实现接口Serializable(该接口仅是一个标记接口,不包括任何方法);
  2. 使用ObjectOutputStream包装一个OutputStream对象,在调用ObjectOutputStream的writeObject()来序列化;
  3. 使用ObjectInputStream包装一个InputStream对象,在调用ObjectInputStream的readObject()来反序列化;

因为序列化最终是字节序列,所以必须字节流,而不能使用字符流(关于字节流和字符流请参考这里)。

对象序列化特别“聪明”的一个地方是它不仅保存了对象的“全景图”,而且能追踪对象内包含的所有引用并保存那些对象;接着又能对每个这样的对象内包含的引用进行追踪;以此类推。这种情况有时被称为“对象网”,单个对象可与之建立连接,而且它还包含了对象的引用数组以和成员对象。如果必须保持一套自己的对象序列化机制,那么维护那些可追踪到所有链接的代码可能会显得非常麻烦。然而,由于 Java的对象序列化似乎找不出什么缺点,所以请尽量不要自己动手,让它用优化的算法自动维护整个对象网。

序列化Demo

public class App
{
public static void main( String[] args ) throws Exception, IOException
{
User user=new User();
user.setName("net.oseye");
user.setAge(18);; //序列化
//使用ObjectOutputStream包装OutputStream
ObjectOutputStream out=new ObjectOutputStream(new FileOutputStream("out.txt"));
out.writeObject("下面是序列化的User:");
out.writeObject(user);
out.close(); //反序列化
//
ObjectInputStream in=new ObjectInputStream(new FileInputStream("out.txt"));
System.out.println(in.readObject());
//查看反序列化后的对象类型,方便反射
Object obj=in.readObject();
System.out.println(obj.getClass()); user=(User)obj;
System.out.println(user);
System.out.println("user的name:"+user.getName());
System.out.println("user的age:"+user.getAge());
in.close();
}
} //实现标记接口Serializable
@SuppressWarnings("serial")
class User implements Serializable{
private String name;
public String getName(){
return name;
}
public void setName(String name){
this.name=name;
} private int age;
public int getAge(){
return age;
}
public void setAge(int age){
this.age=age;
}
}

输出

下面是序列化的User:
class net.oseye.SerializationDemo.User
net.oseye.SerializationDemo.User@9980d5
user的name:net.oseye
user的age:18

如果我们不想序列化某个字段可以使用transient(瞬时)关键字来修饰字段,如:

public class App
{
public static void main( String[] args ) throws Exception, IOException
{
//如前面示例代码....
}
} //实现标记接口Serializable
class User implements Serializable{
private String name;
public String getName(){
return name;
}
public void setName(String name){
this.name=name;
} private transient int age;
public int getAge(){
return age;
}
public void setAge(int age){
this.age=age;
}
}

这样就不会序列化age字段了。

使用Externalizable定制序列化

使用transient关键字可以简单定制序列化,但如果更复杂的定制可以使用Externalizable接口代替标记接口Serializable,因为Externalizable有序列化和反序列化的方法

public class App
{
public static void main( String[] args ) throws Exception, IOException
{
//如前面示例代码....
}
} //实现标记接口Externalizable
class User implements Externalizable{
public User(){}
private String name;
public String getName(){
return name;
}
public void setName(String name){
this.name=name;
} private transient int age;
public int getAge(){
return age;
}
public void setAge(int age){
this.age=age;
} public void writeExternal(ObjectOutput out) throws IOException {
this.name="已经被序列化了:"+this.name;
out.writeObject(this.name);
}
public void readExternal(ObjectInput in) throws IOException,
ClassNotFoundException {
this.name=(String)in.readObject();
}
}

输出

下面是序列化的User:
class net.oseye.SerializationDemo.User
net.oseye.SerializationDemo.User@ff2413
user的name:已经被序列化了:net.oseye
user的age:0

问题:我发现使用Externalizable接口实现的类必须显示声明构造函数,如上例不显示声明

public User(){}

将会报异常:

Exception in thread "main" java.io.InvalidClassException: net.oseye.SerializationDemo.User; no valid constructor

至于原因有待查明....,如果你知道也请告诉我,谢谢。

使用Serializable定制序列化

其实如果你不是特别坚持使用Externalizable接口的话,使用Serializable接口也可以实现定制,但却使用迷惑,因为需要添加

private void writeObject(ObjectOutputStream out) throws IOException;
private void readObject(ObjectInputStream in) throws IOException,
ClassNotFoundException;

这两个方法,如果你细心你会发现和Externalizable接口的中的两个方法很像。迷惑的地方是Serializable是标记接口,而我们感觉无端增加了两个私有方法!!

一旦在接口Serializable的实现类中添加了上面两个方法,序列化和反序列化就会自动调用:

//实现标记接口Serializable
@SuppressWarnings("serial")
class User implements Serializable{
private String name;
public String getName(){
return name;
}
public void setName(String name){
this.name=name;
} private int age;
public int getAge(){
return age;
}
public void setAge(int age){
this.age=age;
} private void writeObject(ObjectOutputStream out) throws IOException {
this.name="已经被序列化了:"+this.name;
out.writeObject(this.name);
}
private void readObject(ObjectInputStream in) throws IOException,
ClassNotFoundException {
this.name=(String)in.readObject();
}
}

Java对象序列化的使用和定制的更多相关文章

  1. 理解Java对象序列化(二)

    关于Java序列化的文章早已是汗牛充栋了,本文是对我个人过往学习,理解及应用Java序列化的一个总结.此文内容涉及Java序列化的基本原理,以及多种方法对序列化形式进行定制.在撰写本文时,既参考了Th ...

  2. java 对象序列化 RMI

    对于一个存在于Java虚拟机中的对象来说,其内部的状态只保持在内存中.JVM停止之后,这些状态就丢失了.在很多情况下,对象的内部状态是需要被持久化下来的.提到持久化,最直接的做法是保存到文件系统或是数 ...

  3. Java对象序列化入门

      Java对象序列化入门 关于Java序列化的文章早已是汗牛充栋了,本文是对我个人过往学习,理解及应用Java序列化的一个总结.此文内容涉及Java序列化的基本原理,以及多种方法对序列化形式进行定制 ...

  4. 深入理解Java对象序列化

    关于Java序列化的文章早已是汗牛充栋了,本文是对我个人过往学习,理解及应用Java序列化的一个总结.此文内容涉及Java序列化的基本原理,以及多种方法对序列化形式进行定制.在撰写本文时,既参考了Th ...

  5. Java对象序列化与RMI

    对于一个存在于Java虚拟机中的对象来说,其内部的状态只保持在内存中.JVM停止之后,这些状态就丢失了.在很多情况下,对象的内部状态是需要被持久化下来的.提到持久化,最直接的做法是保存到文件系统或是数 ...

  6. 理解Java对象序列化【转】

      原文链接:http://www.blogjava.net/jiangshachina/archive/2012/02/13/369898.html 关于Java序列化的文章早已是汗牛充栋了,本文是 ...

  7. Java对象序列化剖析

    对象序列化的目的 1)希望将Java对象持久化在文件中 2)将Java对象用于网络传输 实现方式 如果希望一个类的对象可以被序列化/反序列化,那该类必须实现java.io.Serializable接口 ...

  8. 理解Java对象序列化

    http://www.blogjava.net/jiangshachina/archive/2012/02/13/369898.html 1. 什么是Java对象序列化 Java平台允许我们在内存中创 ...

  9. java 对象序列化与反序列化

    Java序列化与反序列化是什么? 为什么需要序列化与反序列化? 如何实现Java序列化与反序列化? 本文围绕这些问题进行了探讨. 1.Java序列化与反序列化  Java序列化是指把Java对象转换为 ...

随机推荐

  1. 与文件上传到的三个类:FileItem类、ServletFileUpload 类、DiskFileItemFactory类

    文件上传: ServletFileUpload负责处理上传的文件数据,并将表单中每个输入项封装成一个FileItem对象中, 在使用ServletFileUpload对象解析请求时需要根据DiskFi ...

  2. SQL Server 2005入门到精通(案例详解)

    SQL Server 2005基础应用   一.数据库的基本操作 --创建数据库 create database new_db2 on primary ( name='new.mdf', filena ...

  3. 使用原始XML资源——使用原始XML文件

    下面为示例程序添加一个原始的XML文件,将该XML文件放到/res/xml目录下,该XML文件的内容很简单.XML资源的内容如下. 程序清单:  \res\xml\books.xml文件 <?x ...

  4. php生成PDF文件(FPDF)

    首先要下载FPDF http://www.fpdf.org/  附件可以在我的资源里下载http://yunpan.cn/c3RJ5BpPfX6dL  访问密码 f1f2 FPDF文档:http:// ...

  5. 如何去掉Atom的右键菜单?

    Win+R -- regedit -- Ctrl+F -- 搜索"Atom"(或者直接"Open with Atom") -- 找到有个值为Open with ...

  6. Pomelo的Protobuf

    pomelo的protobuf实现,借助了javascript的动态性,使得应用程序可以在运行时解析proto文件,不需要进行proto文件的编译.pomelo的实现中,为了更方便地解析proto文件 ...

  7. Jsoup后台解析html、jsp网页

    在一些网络爬虫或者从第三方网站抓取信息的程序都面临1个问题,如何从网页中把所需的信息提取出来,Jsoup是个比较好的选择,它能把网站内容解析成Document,再从document中取element就 ...

  8. android: activity之间切换的抽屉效果

    之前一直用的是向左平移和向右平移的切换动画,看到别的APP那个抽屉效果,自己也弄了一个!感谢给我提供帮助的大神们! 将退出动画的参数设置为0时,进入动画则设置为向左平移,就实现了抽屉效果! 进入的动画 ...

  9. twemproxyRedis协议解析探索——剖析twemproxy代码正编

    这篇文章会对twemproxyRedis协议解析代码部分进行一番简单的分析,同时给出twemproxy目前支持的所有Redis命令.在这篇文章开始前,我想大家去简单地理解一下有限状态机,当然不理解也是 ...

  10. 雷锋推到雷峰塔,Java implements Javascript。

    最近遇到这么一个问题,如何让用户在软件中自定义函数. 举个例子,使用Java做一个小的监控系统,用户A希望CPU超过90%的时候报警,B用户希望内存超过90%的时候报警,C用户希望CPU超过90%或者 ...