为什么我的序列化文件是 0 字节?为什么它在读取它们时会抛出 EOF 错误?

Why are my serialized files 0 bytes? And why is it throwing an EOF error when reading them?

提问人:Nephty 提问时间:4/10/2021 最后编辑:Mark RotteveelNephty 更新时间:5/2/2021 访问量:462

问:

我正在做一个游戏项目,我正在创建一个保存功能。我正在使用序列化文件,以便直接保存对象并在以后检索它们。我只需要保存一个,其中是一个包含 、 、 、 和 用于特殊上下文的自定义枚举。ArrayList<Direction>DirectionUPDOWNLEFTRIGHTNULLRESTART

在我的代码中,我创建了一个包含用户自游戏开始以来所做的每个动作,并制作了一个保存按钮,将该 ArrayList 存储在文件中。在文件中写入对象似乎没问题,没有抛出错误或其他什么。使用从文件中检索对象的加载按钮时,我收到 EOF 错误(错误写在代码下方)。ArrayList<Direction>

我注意到我的文件是 0 字节,这让我想知道数据是否真的写入了文件中,这可以解释 EOF 错误。我尝试更改文件扩展名,在关闭之前刷新它,更改文件名,但这些似乎都无法解决问题。显然,我需要了解为什么我的文件是 0 字节(从而阻止我检索任何数据)以及为什么我在读取文件时收到 EOF 错误。ObjectOutputStream

以下是使用的代码:

public class LevelSaver {
    public LevelSaver() {
    }

    public void saveLevel(ArrayList<Direction> movesHistory, byte level) throws IOException {
        String fileName = "level" + (level < 10 ? "0" + level : level);
        DateTimeFormatter formatter = DateTimeFormatter.ofPattern("dd-MM-yyyy @ HH-mm-ss");
        String currentTime = LocalDateTime.now().format(formatter);
        fileName += " " + currentTime;
        fileName += ".moves";

        File saveFile = new File("src\\main\\resources\\level\\saves\\" + fileName);

        FileOutputStream fileOutputStream = null;
        ObjectOutputStream objectOutputStream = null;

        try {
            fileOutputStream = new FileOutputStream("src\\main\\resources\\level\\saves" + fileName);
            objectOutputStream = new ObjectOutputStream(fileOutputStream);
            objectOutputStream.writeObject(movesHistory);
            objectOutputStream.flush();
        } catch (IOException exception) {
            exception.printStackTrace();
        }

        saveFile.createNewFile();
    }

    public ArrayList<Direction> getHistory(String fileName) {
        FileInputStream fileInputStream = null;
        ObjectInputStream objectInputStream = null;
        try {
            fileInputStream = new FileInputStream("src\\main\\resources\\level\\saves\\" + fileName);
            objectInputStream = new ObjectInputStream(fileInputStream);
            return (ArrayList<Direction>) objectInputStream.readObject();
        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        } finally {
            if (fileInputStream != null) {
                try {
                    fileInputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (objectInputStream != null) {
                try {
                    objectInputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        return new ArrayList<>();
    }
}

saveLevel()创建一个包含 的文件,并返回存储在文件中的文件。至少他们应该这样做。.movArrayListgetHistory()ArrayList

以下是使用 F 和 G 键分别保存和加载移动历史记录时发生的情况。

case F:
    try {
        levelSaver.saveLevel(movesHistory, currentLevel);
    } catch (IOException e) {
        e.printStackTrace();
    }
    direction = Direction.NULL;
    break;
case G:
    // I'm directly specifying the file and it is intended, the file name is correct
    ArrayList<Direction> res = levelSaver.getHistory("level06 09-04-2021 @ 22-08-57.mov");
    direction = Direction.NULL;
    for (Direction dir : res) {
        System.out.print("applied : " + dir);
        // apply moves applies the given move and changes the displayed map on the screen
        applyMove(dir);
    }
    break;

这是我使用加载功能时遇到的错误。创建 ObjectInputStream 时发生错误。

java.io.EOFException
        at java.base/java.io.ObjectInputStream$PeekInputStream.readFully(ObjectInputStream.java:2932)
        at java.base/java.io.ObjectInputStream$BlockDataInputStream.readShort(ObjectInputStream.java:3427)
        at java.base/java.io.ObjectInputStream.readStreamHeader(ObjectInputStream.java:962)
        at java.base/java.io.ObjectInputStream.<init>(ObjectInputStream.java:405)
        at model.LevelSaver.getHistory(LevelSaver.java:41)
        at view.PlayingMenu$1.handle(PlayingMenu.java:150)
        at view.PlayingMenu$1.handle(PlayingMenu.java:87)
        at javafx.base/com.sun.javafx.event.CompositeEventHandler.dispatchBubblingEvent(CompositeEventHandler.java:86)
        at javafx.base/com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:234)
        at javafx.base/com.sun.javafx.event.EventHandlerManager.dispatchBubblingEvent(EventHandlerManager.java:191)
        at javafx.base/com.sun.javafx.event.CompositeEventDispatcher.dispatchBubblingEvent(CompositeEventDispatcher.java:59)
        at javafx.base/com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:58)
        at javafx.base/com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
        at javafx.base/com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
        at javafx.base/com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
        at javafx.base/com.sun.javafx.event.BasicEventDispatcher.dispatchEvent(BasicEventDispatcher.java:56)
        at javafx.base/com.sun.javafx.event.EventDispatchChainImpl.dispatchEvent(EventDispatchChainImpl.java:114)
        at javafx.base/com.sun.javafx.event.EventUtil.fireEventImpl(EventUtil.java:74)
        at javafx.base/com.sun.javafx.event.EventUtil.fireEvent(EventUtil.java:54)
        at javafx.base/javafx.event.Event.fireEvent(Event.java:198)
        at javafx.graphics/javafx.scene.Scene$KeyHandler.process(Scene.java:4064)
        at javafx.graphics/javafx.scene.Scene.processKeyEvent(Scene.java:2123)
        at javafx.graphics/javafx.scene.Scene$ScenePeerListener.keyEvent(Scene.java:2591)
        at javafx.graphics/com.sun.javafx.tk.quantum.GlassViewEventHandler$KeyEventNotification.run(GlassViewEventHandler.java:217)
        at javafx.graphics/com.sun.javafx.tk.quantum.GlassViewEventHandler$KeyEventNotification.run(GlassViewEventHandler.java:149)
        at java.base/java.security.AccessController.doPrivileged(AccessController.java:391)
        at javafx.graphics/com.sun.javafx.tk.quantum.GlassViewEventHandler.lambda$handleKeyEvent$1(GlassViewEventHandler.java:248)
        at javafx.graphics/com.sun.javafx.tk.quantum.QuantumToolkit.runWithoutRenderLock(QuantumToolkit.java:412)
        at javafx.graphics/com.sun.javafx.tk.quantum.GlassViewEventHandler.handleKeyEvent(GlassViewEventHandler.java:247)
        at javafx.graphics/com.sun.glass.ui.View.handleKeyEvent(View.java:547)
        at javafx.graphics/com.sun.glass.ui.View.notifyKey(View.java:971)
        at javafx.graphics/com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
        at javafx.graphics/com.sun.glass.ui.win.WinApplication.lambda$runLoop$3(WinApplication.java:174)
        at java.base/java.lang.Thread.run(Thread.java:832)
Java 序列化 EOF

评论

0赞 user207421 4/10/2021
这是因为它是 0 长度,它是 0 长度,因为您在写入文件后调用了它。只需删除它即可。注意:Dion 不重复字符串文字。这只是对错别字的邀请。EOFExceptionsaveFile.createNewFile()
0赞 Mark Rotteveel 5/2/2021
@user207421 乍一看,这听起来很有道理,但 File.createNewFile() 应该只创建一个不存在的文件:“当且仅当具有此名称的文件尚不存在时,原子地创建一个由此抽象路径名命名的新空文件。

答:

1赞 Louis Wasserman 4/10/2021 #1

您不会关闭任何一个流。

使用 try-with-resources:

    try (FileOutputStream fileOutputStream = new FileOutputStream("src\\main\\resources\\level\\saves" + fileName);
         ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream)) {
        objectOutputStream.writeObject(movesHistory);
    } catch (IOException exception) {
        exception.printStackTrace();
    }

评论

0赞 user207421 4/10/2021
没错,但这不是文件长度为 0 的原因,并且您还没有回答有关 .EOFException
0赞 Louis Wasserman 4/10/2021
@user207421:是的。如果不关闭文件,实际上不会将任何内容写入文件,如果它的大小为零,它当然会立即被 EOFException。
0赞 user207421 4/10/2021
不。每次写入和刷新数据,标头也自动刷新,这里没有;这是因为构造会尝试读取标头。使用大多数其他流打开长度为零的文件的行为方式不同。writeObject()BufferedOutputStreamEOFExceptionObjectInputStream
0赞 Nephty 4/10/2021
没有必要使用 try-with-resources 关闭流,这对我有用。感谢您的回答,将其标记为最终答案。
0赞 user207421 4/18/2021
@Nephty 如果没有必要,它怎么可能对你有用?问题出在电话上。这个答案没有任何内容,你应该把它标记为正确。事实并非如此。createNewFile()