现在的位置: 首页 > 综合 > 正文

Java的IO操作(三) – 对象的序列化、ObjectInputStream和ObjectOutputStream类

2012年09月21日 ⁄ 综合 ⁄ 共 3040字 ⁄ 字号 评论关闭

由于Java是一种完全面向对象的高级语言,所以在编写程序的时候数据大都存放在对象当中。我们有时会需要将内存中的整个对象都写入到文件中去,然后在适当的时候再从文件中将对象还原至内存。我们可以使用java.io.ObjectInputStream和java.io.ObjectOutputStream类来完成这个任务。

1、对象的序列化(Serialize)

序列化是指将对象的状态信息转换为可以存储或传输的形式的过程。在序列化期间,对象将其当前状态写入到临时或持久性存储区。以后,可以通过从存储区中读取或反序列化对象的状态,重新创建该对象。

2、ObjectInputStream类 和ObjectOutputStream类

如果我们想要序列化一个对象,如我们自定义的User类的对象,那么这个对象必须实现Serializable接口Serializable接口没有任何的抽象方法,实现这个接口仅仅是为了通知编译器这个对象将要被序列化而已。类似的用法还有Cloneable接口,实现这个接口也只是起到通知编译器的作用。

为了演示如何进行对象的序列化,我们先设计一个User类:

package cls;

import java.io.*;

public class User implements Serializable // 实现Serializable接口,仅仅直到标识这个类可被序列化的作用
{
    // 可序列化对象的版本
    private static final long serialVersionUID = 1L;
    
    private String name;
    private int num;
    
    public User(String name,int num)
    {
        this.name = name;
        this.num = num;
    }
    public String getName()
    {
        return name;
    }
    public int getNum()
    {
        return num;
    }
}

注意,seriaVersionUID是指可序列化对象的版本。如果我们没有指定这个版本,那么编译器为自动为实现Serializable接口的类产生一个seriaVersionUID。如果是自动产生的,那么一次如果更改了User类,则自动产生的seriaVersionUID就会不同。在从文件中读回对象时,如果两个对象的seriaVersionUID不同,就会抛出java.io.InvalidClassException。因此如果我们今后不需要改动User类的话,最好自己指定seriaVersionUID,以防止发生该异常。

下面我们使用ObjectInputStream类 和ObjectOutputStream类 向文件中写入3个User对象,追加1个User对象,最后再从文件中读回对象。

package cls;

import java.io.*;
import java.util.*;
import cls.User;

public class ObjectStreamDemo
{
    public static void main(String[] args)
    {
        User[] user = new User[]{new User("dogg",1),new User("catt",2),new User("pigg",3)};
        
        // 向文件中写入对象
        try
        {
            ObjectStreamDemo.writeObj(user,args[0]);
        }
        catch(Exception e)
        {
            System.out.println(e.toString());
        }
        
        // 向文件中追加对象
        try
        {
            // 要追加的对象
            User[] u = new User[]{new User("append1",4),new User("append2",5)};
            
            ObjectStreamDemo.appendObj(u,args[0]);
        }
        catch(Exception e)
        {
            System.out.println(e.toString());
        }
        // 读取对象
        try
        {
            List<User> list = ObjectStreamDemo.readObj(args[0]);
            
            // 输出对象信息
            Iterator<User> it = list.iterator();
            while(it.hasNext())
            {
                User temp = it.next();
                System.out.println(temp.getName() + "," + temp.getNum());
            }
        }
        catch(Exception e)
        {
            System.out.println(e.toString());
        }
    }
    
    static private void appendObj(Object[] objs,String fileName) throws Exception
    {
        File file = new File(fileName);
        
        // 以追加模式创建文件流对象
        FileOutputStream fis = new FileOutputStream(file,true);
        ObjectOutputStream oos = new ObjectOutputStream(fis)
        {
            // 重写 writeStreamHeader()方法,空实现
            protected void writeStreamHeader(){};
        };
        
        // 写入数据
        for(Object o : objs)
        {
            oos.writeObject(o);
        }
        
        // 关闭流
        oos.close();
    }
    static private List<User> readObj(String fileName) throws Exception
    {
        File file = new File(fileName);
        
        // 使用List保存读取出来的对象
        ArrayList<User> list = new ArrayList<User>();
        
        // 创建流对象
        FileInputStream fis = new FileInputStream(file);
        ObjectInputStream ois = new ObjectInputStream(fis);
        
        // 读取对象并放入List容器中
        while(fis.available() > 0)
        {
            list.add((User)ois.readObject());
        }
        
        ois.close();
        return list; // 返回List
    }
    
    static private void writeObj(Object[] objs,String fileName) throws Exception
    {
        // 使用命令行参数中指定的文件名
        File file = new File(fileName);
        
        // 创建流对象
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(file));
        
        // 写入对象
        for(Object o : objs)
        {
            oos.writeObject(o);
        }
        
        // 关闭流
        oos.close();
    }
}

注意,当我们想要向一个已经存在的文件中追加对象时,应该重写ObjectOutputStream的writeStreamHeader()方法,并空实现。因为,ObjectOutputStream在写入数据的时候会加上一个特别的流头(Stream Header),在读取数据的时候会先检查这个流头。所以我们在向文件中追加对象的时候ObjectOutputStream就会再次向文件中写入流头,这样在读取对象的时候会发生StreamCorrupedException异常。

抱歉!评论已关闭.