ObjectOutputStream 追加写入读取错误
Posted LiuYanYGZ
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了ObjectOutputStream 追加写入读取错误相关的知识,希望对你有一定的参考价值。
摘自http://blog.csdn.net/mitkey/article/details/50274543
问题描述:
用类ObjectOutputStream向文件写读对象时,碰到一个问题:新建一个文件,用输出流ObjectOutputStream向文件连续写几个对象,关闭输出流,然 后读取,这些对象都可以读出;这时在向该文件增加对象,新写的对象就读不出了
问题出现的原因:
ObjectOutputStream建立后第一次写入一个对象时, 会在对象数据前写入一些标志的数据“AC ED 00 05”(用二进制方式查看打开),应该是流相关的信息。当你关闭 ObjectOutputStream 后再重新打开往文件里面写对象时(append方式),就会再一次把“AC ED 00 05”写入文件,而这些信息并不是你写入对象的数据,所以当你用ObjectInputStream来读取对象时,流会将除第一个“AC ED 00 05”以外的数据当作各个对象的数据,造成无法解析,所以读取时有一个java.io.StreamCorruptedException出现。 这个可以通过编辑Info.dat来验证,只要将“AC ED 00 05”删除(第一个“AC ED 00 05”保留)就可以正常读出后来加入的对象。 给出一个比较笨的解决方法: 在以后要添加新的对象到Info.dat时,将里面原有的对象读出放入ArrayList中,清空文件,再将对象集一次写入。
尝试解决办法:
那个“AC ED 00 05”是 ObjectOutputStream.writeSystemHeader()写进去的,你可以继承ObjectOutputStream类,覆盖这个方法。 在你自己的writeSystemHeader()里判断是不是第一次写入一个文件,如果是向一个文件大小不为零的文件追加的话,就调用 super.reset(),如果是第一次写这个文件,不是追加,就调用super.writeSystemHeader()
自定义类MyObjectOutputStream
[java] view plain copy- /**
- * 此类继承ObjectOutputStream,重写writeStreamHeader()方法,以实现追加写入时去掉头部信息
- */
- public static class MyObjectOutputStream extends ObjectOutputStream {
- private static File f;
-
- // writeStreamHeader()方法是在ObjectOutputStream的构造方法里调用的
- // 由于覆盖后的writeStreamHeader()方法用到了f。如果直接用此构造方法创建
- // 一个MyObjectOutputStream对象,那么writeStreamHeader()中的f是空指针
- // 因为f还没有初始化。所以这里采用单态模式
- private MyObjectOutputStream(OutputStream out, File f) throws IOException, SecurityException {
- super(out);
- }
-
- // 返回一个MyObjectOutputStream对象,这里保证了new MyObjectOutputStream(out, f)
- // 之前f已经指向一个File对象
- public static MyObjectOutputStream newInstance(File file, OutputStream out) throws IOException {
- f = file;// 本方法最重要的地方:构建文件对象,两个引用指向同一个文件对象
- return new MyObjectOutputStream(out, f);
- }
-
- @Override
- protected void writeStreamHeader() throws IOException {
- // 文件不存在或文件为空,此时是第一次写入文件,所以要把头部信息写入。
- if (!f.exists() || (f.exists() && f.length() == 0)) {
- super.writeStreamHeader();
- } else {
- // 不需要做任何事情
- }
- }
- }
[java] view plain copy
- /**
- * 此类继承ObjectOutputStream,重写writeStreamHeader()方法,以实现追加写入时去掉头部信息
- */
- public static class MyObjectOutputStream extends ObjectOutputStream {
- private static File f;
- // writeStreamHeader()方法是在ObjectOutputStream的构造方法里调用的
- // 由于覆盖后的writeStreamHeader()方法用到了f。如果直接用此构造方法创建
- // 一个MyObjectOutputStream对象,那么writeStreamHeader()中的f是空指针
- // 因为f还没有初始化。所以这里采用单态模式
- private MyObjectOutputStream(OutputStream out, File f) throws IOException, SecurityException {
- super(out);
- }
- // 返回一个MyObjectOutputStream对象,这里保证了new MyObjectOutputStream(out, f)
- // 之前f已经指向一个File对象
- public static MyObjectOutputStream newInstance(File file, OutputStream out) throws IOException {
- f = file;// 本方法最重要的地方:构建文件对象,两个引用指向同一个文件对象
- return new MyObjectOutputStream(out, f);
- }
- @Override
- protected void writeStreamHeader() throws IOException {
- // 文件不存在或文件为空,此时是第一次写入文件,所以要把头部信息写入。
- if (!f.exists() || (f.exists() && f.length() == 0)) {
- super.writeStreamHeader();
- } else {
- // 不需要做任何事情
- }
- }
- }
测试方法:
[java] view plain copy- public static void writeObject() throws Exception {
- String filename = "test.txt";
- File f = new File(filename);
- OutputStream os = new FileOutputStream(f, true);
- MyObjectOutputStream moos = MyObjectOutputStream.newInstance(f, os);
- moos.writeObject("abc");
- moos.writeObject(1);
- moos.close();
- }
[java] view plain copy
- public static void writeObject() throws Exception {
- String filename = "test.txt";
- File f = new File(filename);
- OutputStream os = new FileOutputStream(f, true);
- MyObjectOutputStream moos = MyObjectOutputStream.newInstance(f, os);
- moos.writeObject("abc");
- moos.writeObject(1);
- moos.close();
- }
以上是关于ObjectOutputStream 追加写入读取错误的主要内容,如果未能解决你的问题,请参考以下文章