附加输出流 - Java

Appendable Output Stream - Java

我正在努力将 objects 附加到二进制文件。我的教授提供了一个 "appendable" 输出流 class 供我们在此作业中使用,据我所知,这应该可以防止损坏 header。但是,当我尝试打开二进制文件时,我仍然收到损坏的 header。文件的名称是 test.dat,据我所知,程序可以很好地写入数据,但是一旦我尝试从中读取,所有内容都会消失 window.

fileName是同一个数据字段class这些方法定义在File filename = new File("test.dat");

中,定义如下

如果有人能指出我正确的方向,那就太棒了!提前致谢

我的代码

 /**
 Writes a pet record to the file

 @param pets The pet record to write
 */
 public static void writePets(PetRecord pet){
   AppendObjectOutputStream handle = null;
   try{
     handle = new AppendObjectOutputStream(new FileOutputStream(fileName, true));
     handle.writeObject(pet);
     handle.flush();
   } catch (IOException e){
    System.out.println("Fatal Error!");
    System.exit(0);
   } finally {
   try{
    handle.close();
   } catch (IOException e){
     e.printStackTrace();
   }
 }
}

  /**
  Reads all pets from the file so long as the user continues to enter "next"
  */
  public static void readPets(){
    Scanner keys = new Scanner(System.in);
    String input = "";
    ObjectInputStream handle = null;
    PetRecord pet = null;
    try{
      handle = new ObjectInputStream(new FileInputStream(fileName)); // stack trace points here
      do{
        try{
          pet = (PetRecord) handle.readObject();
          System.out.println("\n" + pet);
          System.out.println("[*] type \"next\" to continue");
          input = keys.nextLine();
        } catch (IOException e){
          System.out.println("\t[*] No More Entries [*]");
          e.printStackTrace();
          break;
        }
      } while (input.matches("^n|^next"));
      handle.close();
    } catch (ClassNotFoundException e){
      System.out.println("The dat file is currupted!");
    } catch (IOException e){
      System.out.println("\t[*] No Entries! [*]");
      e.printStackTrace();
    }
  }

提供class:

public class AppendObjectOutputStream extends ObjectOutputStream
{
   // constructor
   public AppendObjectOutputStream( OutputStream out ) throws IOException
   {
      // this constructor just calls the super (parent)
      super(out);
   }

   @Override
   protected void writeStreamHeader() throws IOException
   {
      // this forces Java to clear the previous header, re-write a new header,
      // and prevents file corruption
      reset();
   }
}

堆栈轨迹:

java.io.StreamCorruptedException: invalid stream header: 79737200
    at java.io.ObjectInputStream.readStreamHeader(ObjectInputStream.java:808)
    at java.io.ObjectInputStream.<init>(ObjectInputStream.java:301)
    at UIHandle.readPets(UIHandle.java:381)
    at UIHandle.list(UIHandle.java:79)
    at UIHandle.command(UIHandle.java:103)
    at UIHandle.mainUI(UIHandle.java:40)
    at UIHandle.main(UIHandle.java:405)

事实证明,如果您确保文件在追加到文件之前退出,它会有所帮助。 问题不在于读取文件,而是在文件不存在时尝试附加到文件。修复是一个简单的 if/else 来检查文件是否存在。如果它不存在则照常写入文件,如果它存在则使用自定义追加 class。

  /**
  Writes a pet record to the file
  @param pet The pet record to write
  */
  public static void writePet(PetRecord pet){
    if (fileName.exists()){
      AppendObjectOutputStream handle = null;
      try{
        handle = new AppendObjectOutputStream(new FileOutputStream(fileName, true));
        handle.writeObject(pet);
        handle.flush();
      } catch (IOException e){
        System.out.println("Fatal Error!");
        System.exit(0);
      } finally {
        try{
          handle.close();
        } catch (IOException e){
          e.printStackTrace();
        }
      }
    } else {
      ObjectOutputStream handle = null;
      try{
        handle = new ObjectOutputStream(new FileOutputStream(fileName));
        handle.writeObject(pet);
        handle.flush();
      } catch (IOException e){
        System.out.println("Fatal Error!");
        System.exit(0);
      } finally {
        try{
          handle.close();
        } catch (IOException e){
          e.printStackTrace();
        }
      }
    }
  }