java 序列化追加文件及反序列化读取多个对象

先来看看java的ObjectOutputStream和ObjectInputStream源码好了

public ObjectOutputStream(OutputStream out) throws IOException {  
        verifySubclass();  
        bout = new BlockDataOutputStream(out);  
        handles = new HandleTable(10, (float) 3.00);  
        subs = new ReplaceTable(10, (float) 3.00);  
        enableOverride = false;  
        writeStreamHeader();  
        bout.setBlockDataMode(true);  
        if (extendedDebugInfo) {  
            debugInfoStack = new DebugTraceInfoStack();  
        } else {  
            debugInfoStack = null;  
        }  
    }  
public ObjectInputStream(InputStream in) throws IOException {  
        verifySubclass();  
        bin = new BlockDataInputStream(in);  
        handles = new HandleTable(10);  
        vlist = new ValidationList();  
        enableOverride = false;  
        readStreamHeader();  
        bin.setBlockDataMode(true);  
    }  

不难发现在序列化new的时候调用了writeStreamHeader()方法写入了4个字节的StreamHeader

但是呢在读取的时候只有读取一次StreamHeader的,所以导致出错

贴图:


这里我们可以看下StreamHeader的内容:

protected void writeStreamHeader() throws IOException {  
        bout.writeShort(STREAM_MAGIC);  
        bout.writeShort(STREAM_VERSION);  
    }  

很简单,一个魔数,一个版本号。两个都是short类型,占4个字节。


解决方法就是把StreamHeader给去掉就行了啦,使用以下方法:

public static void writeObj(File file, Object obj) throws IOException {
		boolean isexit = false;
		if (file.exists()) {
			isexit = true;// 序列化文件存在,追加内容
		}
		FileOutputStream fileOutputStream = new FileOutputStream(file,true);
		// 每次new的时候都会写入一个StreamHeader,所以要把屁股后面的StreamHeader去掉
		ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);
		long pos = 0;// 可以说是文件的长度
		if (isexit) {
			// getChannel()返回此通道的文件位置,这是一个非负整数,它计算从文件的开始到当前位置之间的字节数
			pos = fileOutputStream.getChannel().position() - 4;// StreamHeader有4个字节所以减去
			// 将此通道的文件截取为给定大小
			fileOutputStream.getChannel().truncate(pos);
			System.out.println("追加成功~");
		}

		objectOutputStream.writeObject(obj);
		// 关闭流
		objectOutputStream.close();
		fileOutputStream.close();

	}

(反序列化)读取多个对象:

		FileInputStream fileInputStream = null;
		ObjectInputStream objectInputStream = null;
		java.util.List<Student> list = new ArrayList<Student>();
		try {
			fileInputStream = new FileInputStream(file);
			objectInputStream = new ObjectInputStream(fileInputStream);
			while (fileInputStream.available() > 0) {
				list.add((Student) objectInputStream.readObject());
			}
			System.out.println("读取~");
			for (int i = 0; i < list.size(); i++) {
				System.out.println(list.get(i));
			}

		} catch (FileNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (ClassNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} 





参考博客:点击打开链接


  • 16
    点赞
  • 57
    收藏
    觉得还不错? 一键收藏
  • 7
    评论
序列是将先前序列对象转换为可用的对象的过程。如果你已经将一个对象序列并保存到文件中,现在想要读取文件并还原为原始对象,你需要进行序列操作。在Java中,你可以使用ObjectInputStream类来读取序列后的文件,并将其转换为原始对象。具体步骤如下: 1. 创建一个FileInputStream对象,将序列后的文件作为输入流。 2. 创建一个ObjectInputStream对象,将FileInputStream对象作为输入流。 3. 使用ObjectInputStream对象的readObject()方法读取序列后的对象。该方法返回一个Object对象,你需要将其转换为原始对象类型。 4. 关闭ObjectInputStream和FileInputStream对象。 如果你需要在原始对象中续写一些信息,你可以通过在序列后的文件追加新的序列数据来实现。在Java中,你可以使用ObjectOutputStream类的append()方法来实现此操作。具体步骤如下: 1. 创建一个FileOutputStream对象,将序列后的文件作为输出流。 2. 创建一个ObjectOutputStream对象,将FileOutputStream对象作为输出流。 3. 使用ObjectOutputStream对象的writeObject()方法将新的序列对象写入文件中。 4. 使用ObjectOutputStream对象的flush()方法刷新缓冲区。 5. 关闭ObjectOutputStream和FileOutputStream对象。 在读取包含续写信息的序列文件时,你需要先读取先前序列的内容,然后再读取续写的内容,最后将它们组合成一个完整的对象

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值