提问人:Nephty 提问时间:4/10/2021 最后编辑:Mark RotteveelNephty 更新时间:5/2/2021 访问量:462
为什么我的序列化文件是 0 字节?为什么它在读取它们时会抛出 EOF 错误?
Why are my serialized files 0 bytes? And why is it throwing an EOF error when reading them?
问:
我正在做一个游戏项目,我正在创建一个保存功能。我正在使用序列化文件,以便直接保存对象并在以后检索它们。我只需要保存一个,其中是一个包含 、 、 、 和 用于特殊上下文的自定义枚举。ArrayList<Direction>
Direction
UP
DOWN
LEFT
RIGHT
NULL
RESTART
在我的代码中,我创建了一个包含用户自游戏开始以来所做的每个动作,并制作了一个保存按钮,将该 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()
创建一个包含 的文件,并返回存储在文件中的文件。至少他们应该这样做。.mov
ArrayList
getHistory()
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)
答:
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()
BufferedOutputStream
EOFException
ObjectInputStream
0赞
Nephty
4/10/2021
没有必要使用 try-with-resources 关闭流,这对我有用。感谢您的回答,将其标记为最终答案。
0赞
user207421
4/18/2021
@Nephty 如果没有必要,它怎么可能对你有用?问题出在电话上。这个答案没有任何内容,你应该把它标记为正确。事实并非如此。createNewFile()
评论
EOFException
saveFile.createNewFile()
File.createNewFile()
应该只创建一个不存在的文件:“当且仅当具有此名称的文件尚不存在时,原子地创建一个由此抽象路径名命名的新空文件。