如何创建任意大小的黑白图像

How to create a black and white image of arbitrary size

我想创建一个仅由黑白像素组成的任意大图像。我目前正在使用 BufferedImage。然而,这达到了 65500 像素的硬性限制。

Exception in thread "main" javax.imageio.IIOException: Maximum supported image dimension is 65500 pixels
    at java.desktop/com.sun.imageio.plugins.jpeg.JPEGImageWriter.writeImage(Native Method)
    at java.desktop/com.sun.imageio.plugins.jpeg.JPEGImageWriter.writeOnThread(JPEGImageWriter.java:1007)
    at java.desktop/com.sun.imageio.plugins.jpeg.JPEGImageWriter.write(JPEGImageWriter.java:371)
    at java.desktop/javax.imageio.ImageWriter.write(ImageWriter.java:613)
    at java.desktop/javax.imageio.ImageIO.doWrite(ImageIO.java:1628)
    at java.desktop/javax.imageio.ImageIO.write(ImageIO.java:1554)

如何创建任意尺寸的图像?

加分项:由于图像只是黑白的,因此高效的数据格式会很棒。

您只达到了特定图像编写器的限制。当我尝试使用更适合真正的黑白图像的 PNG 时,可以写入和读回更大的图像。

我使用了以下代码进行测试:

public static void main(String[] args) throws IOException {
    int[][] sizes = {{4000, 4000 }, {40_000, 40_000 }, {70_000, 4000 }, {700_000, 400 }};
    for(int[] size: sizes) {
        int width = size[0], height = size[1];
        System.out.println("trying " + width + " x " + height);
        BufferedImage bi=new BufferedImage(width,height,BufferedImage.TYPE_BYTE_BINARY);
        Graphics2D gfx = bi.createGraphics();
        gfx.setColor(Color.WHITE);
        gfx.fillRect(0, 0, width, height);
        gfx.setColor(Color.BLACK);
        gfx.drawOval(20, 20, width - 40 , height - 40);
        int off = Math.min(width, height) >>> 3;
        gfx.drawOval(off, off, width - off - off, height - off - off);
        gfx.dispose();

        test(bi, "PNG");
        test(bi, "GIF");
        test(bi, "JPEG");
    }
}
static void test(BufferedImage bi, String format) throws IOException {
    Path f = Files.createTempFile("test", "-out."+format);
    try {
        try { ImageIO.write(bi, format, f.toFile()); }
        catch(Throwable t) {
            System.err.println(format+": "+t);
            return;
        }
        BufferedImage read;
        try { read = ImageIO.read(f.toFile()); }
        catch(Throwable t) {
            System.err.println(format+": written, re-reading: "+t);
            return;
        }
        if(bi.getWidth() != read.getWidth() || bi.getHeight() != read.getHeight()) {
            System.err.println(format+": size changed to "
                              +read.getWidth() + " x " + read.getHeight());
            return;
        }
        if(format.equalsIgnoreCase("jpeg")) {
            // turn back to black & white, rounding away JPEG artifacts
            BufferedImage bw = new BufferedImage(
                read.getWidth(), read.getHeight(), BufferedImage.TYPE_BYTE_BINARY);
            Graphics2D gfx = bw.createGraphics();
            gfx.drawImage(read, 0, 0, null);
            gfx.dispose();
            read = bw;
        }
        for(int x = 0, w = bi.getWidth(); x < w; x++) {
            for(int y = 0, h = bi.getHeight(); y < h; y++) {
                if(bi.getRGB(x, y) != read.getRGB(x, y)) {
                    System.err.println(format+": content changed");
                    return;
                }
            }
        }
        System.out.println(format + ": sucessfully written and restored");
    }
    finally {
        Files.deleteIfExists(f);
    }
}

结果:

trying 4000 x 4000
PNG: sucessfully written and restored
GIF: sucessfully written and restored
JPEG: sucessfully written and restored
trying 40000 x 40000
PNG: sucessfully written and restored
GIF: sucessfully written and restored
JPEG: written, re-reading: java.lang.IllegalArgumentException: Invalid scanline stride
trying 70000 x 4000
PNG: sucessfully written and restored
GIF: size changed to 4464 x 4000
JPEG: javax.imageio.IIOException: Maximum supported image dimension is 65500 pixels
trying 700000 x 400
PNG: sucessfully written and restored
GIF: size changed to 44640 x 400
JPEG: javax.imageio.IIOException: Maximum supported image dimension is 65500 pixels

我无法创建宽度×高度超过 int 值范围的 BufferedImage 对象,我想。这种限制似乎在实施的几个地方被硬编码了。

pbm 可能是适合您的图像格式,它甚至不需要使用 Java 的图像库来处理。

static void writeRandomBlackWhiteImage(
        OutputStream out, long width, long height) throws IOException {
    String header = String.format("P4\n%d\n%d\n", width, height);
    out.write(header.getBytes(StandardCharsets.US_ASCII));

    // bytes = (width * height + 7) / 8, throwing ArithmeticException on overflow
    long bytes = Math.addExact(Math.multiplyExact(width, height), 7L) / 8L;
    while (bytes > 0) {
        byte[] data = new byte[(int) Math.min(0x4000, bytes)];
        ThreadLocalRandom.current().nextBytes(data);
        out.write(data);
        bytes -= data.size;
    }
}

public static void main(String[] args) throws IOException {
    try (OutputStream out = new FileOutputStream("out.pbm")) {
        writeRandomBlackWhiteImage(out, 65536, 65536);
    }
}

请注意(与 java.awt.image.BufferedImage 不同)这可以轻松生成尺寸大于 Integer.MAX_VALUE 的图像。 (尽管处理这些图像可能具有挑战性。)