如何在应用程序终止之前将映像保存到磁盘?

时间:2017-10-11 17:16:48

标签: java file javax.imageio

我需要将一个像素数组写入磁盘并在同一个应用程序中读取该文件。出于某种原因,文件不会写入磁盘,直到应用程序终止。 (只有这样它们才会出现在他们保存的目录中)。我在IntelliJ IDEA中编写此应用程序,如果知道这有用的话。

如何确保文件立即写入磁盘?这是我的代码:

protected void savePixelstoPNG(int[] pixels, String fileName) {
    BufferedImage image = new BufferedImage(getMapWidth(), getMapHeight(), BufferedImage.TYPE_INT_RGB);
    Graphics graphics = image.getGraphics();

    for(int y = 0; y < getMapHeight(); y++) {
        for(int x = 0; x < getMapWidth(); x++) {
            graphics.setColor(new Color(pixels[x + y * getMapWidth()]));
            graphics.fillRect(x, y, 1, 1);
        }
    }

    try {
        File file = new File(fileName);
        ImageIO.write(image, "PNG", file);
    } catch(IOException e) {
        e.printStackTrace();
    }
}
编辑:我检查了文件夹,实际上他们正在立即写入磁盘。但是,这些更改不会反映在项目目录中(文件保存到java包中),直到应用程序终止。因此,当我在保存它们之后(在相同的应用程序生命周期内)读取这些文件时,即使它们存在于磁盘上,应用程序也无法找到这些文件。

编辑2:这是我使用相对目录路径从类路径中读取文件的代码。从类路径中读取初始资源。当它们被更新时,它们被写入类路径中的不同目录,因此原始资源不会被覆盖,因为每次重新运行应用程序时都应该首先读取原始资源:

void myLoadMethod() {
        loadMapTiles("resource/tilemap_1-1.png");
        loadTriggerTiles("resource/triggermap_1-1.png");
}


protected void loadMapTiles(@NotNull String path) {
        URL url = getClass().getClassLoader().getResource(path);
        loadTiles(url, mapTiles);
}

protected void loadTriggerTiles(@NotNull String path) {
        URL url = getClaass().getClassLoader().getResource(path);
        loadTiles(url, triggerTiles);
}

protected void loadTiles(@NotNull URL url, @Nullable int[] dest) {
        try {
            System.out.println("Trying to load: " + url.toString() + "...");
            BufferedImage map = ImageIO.read(url);
            int[] pixels = new int[mapWidth * mapHeight];
            map.getRGB(0, 0, mapWidth, mapHeight, pixels, 0, mapWidth);
            System.arraycopy(pixels, 0, dest, 0, dest.length);
            System.out.println("Success!");
        } catch (IOException e) {
            System.out.println("failed...");
            e.printStackTrace();
        }
    }
}

请注意,mapTilestriggerTiles是包含loadMapTiles loadTriggerTilesloadTiles

的类中包含的字段

1 个答案:

答案 0 :(得分:1)

您应该区分源文件和类路径。

源通常驻留在某种src目录中。例如。对于Maven项目,src/main/java文件的标准为*.java,其他资源src/main/resources的标准为*.properties等。

但是后来的java文件被编译,*.class文件被复制到某个输出目录。例如。对于Maven,这是target/...,许多其他工具都有binoutput作为已编译的类文件夹。资源保存在同一位置。

您似乎正在将文件写入src并尝试从其他位置读取它们(例如,Class#getResourceAsStream(path)将从类路径读取数据)。您可以在下一个应用程序运行时读取该文件这一事实使我认为在下一个版本中它会被复制到输出目录并对您的应用程序“可见”。

在你的代码上

URL url = getClass().getClassLoader().getResource(path);

构造一个预期在类路径上的文件的URL(最有可能在${projectRoot}/target/classes/${path}中)。

虽然

File file = new File(fileName);

在相对于执行目录的路径中创建一个文件(很可能是相对于${projectRoot})。

通常ClassLoader#getResource(path)只应用于获取类路径上的静态资源。

要在磁盘上构建路径,请使用Paths#get(path)。如果你绝对需要一个URL,你可以这样做:

Paths.get(path).toURI().toURL();