从图像中剥离Alpha通道

时间:2016-03-16 08:30:02

标签: java bufferedimage alpha twelvemonkeys

我想从PNG中剥离Alpha通道(透明背景),然后将它们写为JPEG图像。更准确地说,我想将透明像素设为白色。我尝试了两种技术,两种技术都以不同的方式失败:

方法1:

BufferedImage rgbCopy = new BufferedImage(inputImage.getWidth(), inputImage.getHeight(), BufferedImage.TYPE_INT_RGB);
Graphics2D graphics = rgbCopy.createGraphics();
graphics.drawImage(inputImage, 0, 0, Color.WHITE, null);
graphics.dispose();
return rgbCopy;

结果:图像呈粉红色背景。

方法2:

final WritableRaster raster = inputImage.getRaster();
final WritableRaster newRaster = raster.createWritableChild(0, 0, inputImage.getWidth(), inputImage.getHeight(), 0, 0, new int[]{0, 1, 2});
ColorModel newCM = new ComponentColorModel(inputImage.getColorModel().getColorSpace(), false, false, Transparency.OPAQUE, DataBuffer.TYPE_BYTE);
return new BufferedImage(newCM, newRaster, false, null);

结果:图片背景为黑色。

在这两种情况下,输入图像都是PNG,输出图像按JPEG编写如下:ImageIO.write(bufferedImage, "jpg", buffer)。如果它是相关的:这是在Java 8上,我使用12monkeys库来重新调整图像,然后再将其写为JPEG。

我已尝试过上述代码的多种变体,但没有运气。以前的一些问题提出了上述代码,但在这种情况下它似乎并没有起作用。

2 个答案:

答案 0 :(得分:6)

感谢Rob的回答,我们现在知道为什么颜色混乱了。

问题是双重的:

  • JPEGImageWriter用于编写JPEG的默认ImageIO,不会以其他软件理解的方式编写带有alpha的JPEG(这是一个已知问题)。
  • null作为目标传递给ResampleOp.filter(src, dest)且过滤方法为FILTER_TRIANGLE时,将创建一个新的BufferedImage,其中包含alpha(实际上为BufferedImage.TYPE_INT_ARGB })。

重新取样后剥离alpha将起作用。但是,还有另一种方法可能更快并节省一些内存。也就是说,不是传递null目的地,而是传递适当大小的BufferedImage并输入:

public static void main(String[] args) throws IOException {
    // Read input
    File input = new File(args[0]);
    BufferedImage inputImage = ImageIO.read(input);

    // Make any transparent parts white
    if (inputImage.getTransparency() == Transparency.TRANSLUCENT) {
        // NOTE: For BITMASK images, the color model is likely IndexColorModel,
        // and this model will contain the "real" color of the transparent parts
        // which is likely a better fit than unconditionally setting it to white.

        // Fill background  with white
        Graphics2D graphics = inputImage.createGraphics();
        try {
            graphics.setComposite(AlphaComposite.DstOver); // Set composite rules to paint "behind"
            graphics.setPaint(Color.WHITE);
            graphics.fillRect(0, 0, inputImage.getWidth(), inputImage.getHeight());
        }
        finally {
            graphics.dispose();
        }
    }

    // Resample to fixed size
    int width = 100;
    int height = 100;

    BufferedImageOp resampler = new ResampleOp(width, height, ResampleOp.FILTER_TRIANGLE);

    // Using explicit destination, resizedImg will be of TYPE_INT_RGB
    BufferedImage resizedImg = resampler.filter(inputImage, new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB));

    // Write output as JPEG
    ImageIO.write(resizedImg, "JPEG", new File(input.getParent(), input.getName().replace('.', '_') + ".jpg"));
}

答案 1 :(得分:1)

自己回答这个问题,因为这可能会帮助其他遇到此问题的人。 haraldk是正确的,还有另一段代码导致alpha通道问题。方法1中的代码是正确的并且有效。

然而。

如果在剥离Alpha通道后,使用十二个键ResampleOp调整图像大小,如下所示:

BufferedImageOp resampler = new ResampleOp(width, height, ResampleOp.FILTER_TRIANGLE);
BufferedImage resizedImg = resampler.filter(rgbImage, null);

这会导致Alpha通道变为粉红色。

解决方案是在剥离Alpha通道之前调整图像的大小。