`

Java序列化与反序列化

    博客分类:
  • Java
阅读更多

在JAVA中,一个大的应用程序需要保存很多对象的时候,由于虚拟机内存有限,(资源宝贵啊 )有时不可能所有有用的对象都放到内存中,因此,需要将不常用的对象暂时持久化的文件中,当需要这个对象时,再从文件把对象恢复到内存中,这就是所谓对象的序列化和反序列化。本文讲实现如何将对象序列化到文件,然后再从文件反序列化到对象,你会发现其实特别简单



先看一下对象的序列化和反序列化的关键是什么

1,首先被序列化的对象必须实现 java.io.Serializable 接口,咳~~咳~~,其实这个接口没定义任何方法



2,序列化时,需要用到对象输出流ObjectOutputStream ,然后通过文件输出流构造 ObjectOutputStream 对象调用writeObject写入到文件



3,反之,反序列化时用到对象输入流ObjectIntputStream, 然后通过文件输出流构造 ObjectIntputStream对象调用readObject加载到内存,注意,是返回万能Object类型



4,注意:序列化时transient变量(这个关键字的作用就是告知JAVA我不可以被序列化)和静态变量不会被序列化(关于静态变量不会序列化保留意见,本实例先不用静态变量)



5,也是最应该注意的,如果你先序列化对象A后序列化B,那么在反序列化的时候一定记着JAVA规定先读到的对象是先被序列化的对象,不要先接收对象B,那样会报错



好了,现在我们做一个实例,我们现在要序列化一个会员对象到文件里,然后再把它读出来

先创建会员类

import java.io.*;   
import java.util.*;   
  
//一定要实现Serializable接口才能被序列化   
public class UserInfo implements Serializable {   
    public String userName;   
    public String userPass;   
    //注意,userAge变量前面的transient   
    public transient int userAge;   
  
    public UserInfo(){   
    }   
  
    public UserInfo(String username,String userpass,int userage){   
        this.userName=username;   
        this.userPass=userpass;   
        this.userAge=userage;   
    }   
  
    public String toString(){   
        return "用户名: "+this.userName+";密码:"+this.userPass+   
            ";年龄:"+this.userAge;   
    }   
}  

 

接着我们开始写如何序列化和反序列化,初学认真看哦

import java.io.*;   
import java.util.*;   
public class Test {   
  
    //序列化对象到文件   
    public static void serialize(String fileName){   
        try  
        {   
            //创建一个对象输出流,讲对象输出到文件   
            ObjectOutputStream out=new ObjectOutputStream(new FileOutputStream(fileName));   
  
            out.writeObject("序列化日期是:");  //序列化一个字符串到文件   
  
            out.writeObject(new Date());    //序列化一个当前日期对象到文件   
  
            UserInfo user=new UserInfo("renyanwei","888888",20);   
            out.writeObject(user);  //序列化一个会员对象   
  
            out.close();   
        }   
        catch (Exception x)   
        {   
            System.out.println(x.toString());   
        }   
           
    }   
    //从文件反序列化到对象   
    public static void deserialize(String fileName){   
        try  
        {   
            //创建一个对象输入流,从文件读取对象   
            ObjectInputStream in=new ObjectInputStream(new FileInputStream(fileName));   
  
            //注意读对象时必须按照序列化对象顺序读,否则会出错   
            //读取字符串   
            String today=(String)(in.readObject());   
            System.out.println(today);   
  
            //读取日期对象   
            Date date=(Date)(in.readObject());   
            System.out.println(date.toString());   
  
            //读取UserInfo对象并调用它的toString()方法   
            UserInfo user=(UserInfo)(in.readObject());             
            System.out.println(user.toString());   
  
            in.close();   
        }   
        catch (Exception x)   
        {   
            System.out.println(x.toString());   
        }   
           
    }   
  
    public static void main(String[] args) {       
        serialize("D:\\test.txt");   
        System.out.println("序列化完毕");   
  
        deserialize("D:\\test.txt");   
        System.out.println("反序列化完毕");   
    }   
  
} 

运行结果:


序列化完毕
今天是:
Thu Oct 23 18:31:11 CST 2008
用户名: renyanwei;密码:888888;年龄:0
反序列化完毕

我们看代码总结一下,serialize方法实现了对象的序列化功能,讲对象序列化到文件,使用了文件的输出流ObjectOutputStream 的writerObject方法一次写入一个对象,对象在文件中按写入顺序存储。

deserialize方法实现反序列化,从文件将对象反序列化到内存,使用了ObjectIntputStream 的readObject方法,因为序列化对象是按照写入顺序存储的,所以在反序列化时必须按照写入顺序读取

大家看了结果就应该知道了,标明transit的实例变量是没被序列化进去的 ,呵呵



其实序列化的意义不止如此,不但可以将对象放到文件流,还可以放到网络流里,比如我们做的Socket程序,你不但可以发送给对方一行消息,还可以发送过去一个对象,我估计QQ就不止是把我们的消息发送到对方,应该是一个对象,里面包含了发送者的信息

接下来这次我们讲的是控制对象的序列化和反序列化



控制序列化就是有选择的序列化对象,而不是把对象的所以内容都序列化,前篇我们的例子中介绍了transit变量和类变量(static)不被序列化,现在我们还有一种更为灵活的控制对象序列化和反序列方法,可以在序列化过程中储存其他非this对象包含的数据



我们现在再来介绍一个接口 java.io.Externalizable 。当对象实现了这个接口时,就可以灵活的控制它的序列化和反序列过程,这个接口继承了java.io.Serializable



Externalizable 接口定义了两个方法,writerExternal方法在序列化时被调用,可以再该方法中控制序列化内容,readExternal方法在反序列时被调用,可以在该方法中控制反序列的内容



好了,现在我们改写上例的UserInfo

import java.io.*;   
import java.util.*;   
  
//本程序通过实现Externalizable接口控制对象序列化和反序列   
public class UserInfo implements Externalizable {   
    public String userName;   
    public String userPass;   
    public int userAge;   
  
    public UserInfo(){   
    }   
  
    public UserInfo(String username,String userpass,int userage){   
        this.userName=username;   
        this.userPass=userpass;   
        this.userAge=userage;   
    }   
       
    //当序列化对象时,该方法自动调用   
    public void writeExternal(ObjectOutput out) throws IOException{   
        System.out.println("现在执行序列化方法");   
        //可以在序列化时写非自身的变量   
        Date d=new Date();   
        out.writeObject(d);   
        //只序列化userName,userPass变量   
        out.writeObject(userName);   
        out.writeObject(userPass);   
    }   
  
    //当反序列化对象时,该方法自动调用   
    public void readExternal(ObjectInput in) throws IOException,ClassNotFoundException{   
        System.out.println("现在执行反序列化方法");   
        Date d=(Date)in.readObject();   
        System.out.println(d);   
        this.userName=(String)in.readObject();   
        this.userPass=(String)in.readObject();   
    }   
  
    public String toString(){   
        return "用户名: "+this.userName+";密码:"+this.userPass+   
            ";年龄:"+this.userAge;   
    }   
} 

改写测试类 

 

import java.io.ObjectOutputStream;   
import java.io.ObjectInputStream;   
import java.io.FileInputStream;   
import java.io.FileOutputStream;   
import java.util.Date;   
import java.lang.management.*;   
public class Test {   
  
  
  
    //序列化对象到文件   
    public static void serialize(String fileName){   
        try  
        {   
            //创建一个对象输出流,讲对象输出到文件   
            ObjectOutputStream out=new ObjectOutputStream(new FileOutputStream(fileName));   
  
            UserInfo user=new UserInfo("renyanwei","888888",20);   
            out.writeObject(user);  //序列化一个会员对象   
  
            out.close();   
        }   
        catch (Exception x)   
        {   
            System.out.println(x.toString());   
        }   
           
    }   
    //从文件反序列化到对象   
    public static void deserialize(String fileName){   
        try  
        {   
            //创建一个对象输入流,从文件读取对象   
            ObjectInputStream in=new ObjectInputStream(new FileInputStream(fileName));   
  
            //读取UserInfo对象并调用它的toString()方法   
            UserInfo user=(UserInfo)(in.readObject());             
            System.out.println(user.toString());   
  
            in.close();   
        }   
        catch (Exception x)   
        {   
            System.out.println(x.toString());   
        }   
           
    }   
  
    public static void main(String[] args) {       
  
        serialize("D:\\test.txt");   
        System.out.println("序列化完毕");   
           
        deserialize("D:\\test.txt");   
        System.out.println("反序列化完毕");   
    }   
  
} 

执行结果

现在执行序列化方法
序列化完毕
现在执行反序列化方法
Thu Oct 23 22:18:12 CST 2008
用户名: renyanwei;密码:888888;年龄:0
反序列化完毕

首先,我们在序列化UserInfo对象的时候,由于这个类实现了Externalizable 接口,在writeExternal()方法里定义了哪些属性可以序列化,哪些不可以序列化,所以,对象在经过这里就把规定能被序列化的序列化保存文件,不能序列化的不处理,然后在反序列的时候自动调用readExternal()方法,根据序列顺序挨个读取进行反序列,并自动封装成对象返回,然后在测试类接收,就完成了反序列

分享到:
评论

相关推荐

    java序列化与反序列化

    java序列化与反序列化

    探索Java序列化与反序列化的奥秘:数据持久化的桥梁

    ### Java序列化与反序列化的深入解析 #### Java序列化的重要性及应用场景 Java序列化是一项核心功能,它允许程序员将对象的状态转化为字节流的形式,从而实现对象的持久化存储或者在网络之间进行传递。这对于诸如...

    Java序列化与反序列化<转>

    Java序列化与反序列化是Java编程中一个重要的概念,它允许我们将对象的状态转换为字节流,以便存储或在网络中传输。这个过程被称为序列化,而将字节流恢复为对象的过程则称为反序列化。Java提供了一个内置的机制来...

    《Java 序列化与反序列化:基础与应用》.docx

    《Java 序列化与反序列化:基础与应用》

    java序列化和反序列化

    ### Java序列化与反序列化详解 #### 一、Java序列化概述 Java序列化(Serialization)是一项重要的功能,它可以将对象的状态转化为一系列字节,从而实现对象的持久化存储或在网络上传输。序列化机制使得Java对象...

    java序列化和反序列化的方法

    java 序列化和反序列化的方法 Java 序列化和反序列化是 Java 语言中的一种机制,用于将对象转换为字节流,以便在网络上传输或存储。序列化是将对象转换为字节流的过程,而反序列化是将字节流转换回对象的过程。 在...

    java 对象的序列化与反序列化

    Java对象的序列化和反序列化是Java编程中一项重要的技术,主要用于将对象的状态转换为字节流,以便存储或在网络上传输。这一过程对于理解Java的IO操作、持久化数据以及实现分布式通信等场景非常关键。 首先,我们来...

    Java序列化和反序列化

    ### Java序列化与反序列化的深入解析 #### 序列化的功能与意义 序列化是Java编程语言中的一项核心功能,其主要目的是将对象的状态转换为可以存储或传输的格式,便于持久化保存或网络传输。序列化并不涉及对象的...

    java 序列化与反序列化的实例详解

    java 序列化与反序列化的实例详解 Java 序列化与反序列化是 Java 编程语言中的一种机制,允许将 Java 对象转换为字节序列,并将其反序列化回 Java 对象。序列化是指将 Java 对象转换为字节序列的过程,而反序列化是...

    java serializable 序列化与反序列化

    Java的序列化与反序列化是Java开发中的一项重要技术,它允许我们将对象的状态转换为字节流,以便存储或在网络上传输。`Serializable`接口是Java提供的一个标记接口,用于实现对象的序列化。当一个类实现了这个接口,...

    java序列化(Serializable)的作用和反序列化.doc

    ### Java序列化(Serializable)的作用与反序列化详解 #### 一、序列化的概念 序列化是指将程序中的对象转换为一系列字节流的过程,主要用于保存对象的状态或在网络之间传输对象。序列化的主要目的是为了能够持久化...

    java反序列化工具

    Java反序列化是一种将已序列化的对象状态转换回对象的过程,它是Java平台中持久化数据的一种常见方式。在Java应用程序中,序列化用于保存对象的状态以便稍后恢复,或者在网络间传输对象。然而,这个过程也可能引入...

    Java对象序列化和反序列化工具Xson.zip

    Xson是一个Java对象序列化和反序列化程序。支持Java对象到字节数组的序列化,和从字节数组到Java对象的反序列化。 Maven:  &lt;groupId&gt;com.github.xsonorg&lt;/groupId&gt;  &lt;artifactId&gt;xson-core  &lt;version&gt;1.0.1 ...

    C#和Java的序列化反序列化

    本篇文章将深入探讨C#和Java中的序列化与反序列化机制。 首先,我们要了解什么是序列化。序列化是指将对象的状态转化为可存储或可传输的数据格式的过程。这个过程通常将内存中的对象转换成字节流,以便保存到磁盘、...

    Java Json序列化与反序列化

    Java中的JSON(JavaScript Object Notation)序列化与反序列化是开发过程中常见且重要的操作,主要用于数据交换和存储。JSON是一种轻量级的数据交换格式,它允许我们将Java对象转换为JSON字符串,反之亦然,方便在...

    详解Java 序列化与反序列化(Serialization)

    Java 序列化与反序列化(Serialization)知识点总结 Java 序列化与反序列化是 Java 语言中的一种机制,用于将对象的状态信息转化为可以存储或者传输的形式的过程。序列化(Serialization)是将对象的状态信息转化为...

Global site tag (gtag.js) - Google Analytics