【问题标题】:Java - Can't write object as file to directoryJava - 无法将对象作为文件写入目录
【发布时间】:2016-10-07 15:00:51
【问题描述】:

我的程序有一个奇怪的问题。 我写了一个小程序,将对象写入特定目录中的文件。

玩家等级:

package readwrite;

import java.io.Serializable;

public class Player implements Serializable {

    private String name;
    private int level;

    public Player(String name, int level) {
        this.name = name;
        this.level = level;
    }

    public String getName() {
        return name;
    }

    public int getLevel() {
        return level;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void setLevel(int level) {
        this.level = level;
    }

    public String toString() {
        return "Name: " + name + ", Level: " + level;
    }

    public void printInfo() {
        System.out.println(this.toString());
    }

}

PlayerReaderWriter-类:

package readwrite;

import java.io.*;

public class PlayerReaderWriter {

    public static Player readPlayer(String path) {
        Player p = null;

        try {
            FileInputStream fileIn = new FileInputStream(path);
            ObjectInputStream in = new ObjectInputStream(fileIn);
            p = (Player) in.readObject();
            in.close();
            fileIn.close();
        } catch(IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }

        return p;
    }

    public static void writePlayer(String path, Player p) {
        try {
            File file = new File(path);
            if (file.exists()) {
                file.delete();
                file.createNewFile();
            } else {
                file.mkdirs();
                file.createNewFile();
            }

            FileOutputStream fileOut = new FileOutputStream(path);
            ObjectOutputStream out = new ObjectOutputStream(fileOut);
            out.writeObject(p);
            out.close();
            fileOut.close();
        } catch(IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        Player p = new Player("Foo", 64);
        p.printInfo();
        writePlayer("players/Player.ser", p);

        Player q = readPlayer("players/Player.ser");
        q.printInfo();
    }

}

问题是:当我第一次运行我的程序时,当目录“玩家”不存在时,我收到以下错误:

D:\Projects>java readwrite/PlayerReaderWriter
Name: Foo, Level: 64
java.io.FileNotFoundException: players\Player.ser (Access is denied)
        at java.io.FileOutputStream.open0(Native Method)
        at java.io.FileOutputStream.open(FileOutputStream.java:270)
        at java.io.FileOutputStream.<init>(FileOutputStream.java:213)
        at java.io.FileOutputStream.<init>(FileOutputStream.java:101)
        at readwrite.PlayerReaderWriter.writePlayer(PlayerReaderWriter.java:34)
        at readwrite.PlayerReaderWriter.main(PlayerReaderWriter.java:47)
java.io.FileNotFoundException: players\Player.ser (Access is deniedt)
        at java.io.FileInputStream.open0(Native Method)
        at java.io.FileInputStream.open(FileInputStream.java:195)
        at java.io.FileInputStream.<init>(FileInputStream.java:138)
        at java.io.FileInputStream.<init>(FileInputStream.java:93)
        at readwrite.PlayerReaderWriter.readPlayer(PlayerReaderWriter.java:11)
        at readwrite.PlayerReaderWriter.main(PlayerReaderWriter.java:49)
Exception in thread "main" java.lang.NullPointerException
        at readwrite.PlayerReaderWriter.main(PlayerReaderWriter.java:50)

现在程序已经创建了“players”目录,并且由于某种原因它还创建了一个名为“Player.ser”的子目录

当我第二次运行它时,它会工作并在“players”目录中创建一个“Player.ser”文件

当我只用一个名为“players”的空目录运行它时,它也不起作用。

问候:)

【问题讨论】:

  • 我相信你会得到你想要的。 :-) 第一次,该文件不存在,所以您的程序执行 file.mkdirs() — 这将创建目录播放器和 目录 Player.ser。下一次 Player.ser 目录存在(你没有问它是否是一个目录,只问它是否存在),所以你的程序删除目录并创建一个文件。只有一个空的玩家目录,你会得到第一个行为,除了玩家目录没有创建,因为它已经存在,其余的都是一样的。顺便说一句,您应该能够在一个更小的示例中重现该行为。 :-)

标签: java file object io directory


【解决方案1】:

看这段代码:

if (file.exists()) {
    file.delete();
    file.createNewFile();
} else {
    file.mkdirs();
    file.createNewFile();
}

您是说如果players/Player.ser 不存在,您应该将其创建为目录 - 然后也将其创建为文件。坏了。

如果该文件已经存在,则根本不需要删除它 - 您可以使用 FileOutputStream 并让它覆盖该文件,并在开始时截断它,这是默认行为。

你只需要先检查目录是否存在,例如

File file = new File(path);
File parent = file.getParentFile();
// TODO: Handle parent existing, but not being a directory,
// or file existing, but being a directory...
if (!parent.exists()) {
    parent.mkdirs();
}

try (OutputStream output = new FileOutputStream(file)) {
    // ...
}

请注意 try-with-resources 语句,以便您的输出流始终关闭,即使抛出异常也是如此。

【讨论】:

  • getParentFile() 而不是 getParent() :)
【解决方案2】:

查看File#mkdirs() 上的 JavaDoc:

创建由 this 抽象路径名命名的目录...

这表示该文件被假定为一个目录,因此Player.ser 被创建为一个目录。请改用file.getParentFile().mkdirs()(请记住,根据路径可能没有父级)。

由于您的代码正在创建目录,因此对该 File 对象的任何写入操作都会失败,因为您无法将数据直接写入目录。

【讨论】:

    猜你喜欢
    • 2016-02-29
    • 2018-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-08-18
    相关资源
    最近更新 更多