如评论中所述,您拥有的代码应该可以工作,除非您的 AdobeRGB1998 ICC 配置文件有什么特别之处......
下面的代码对我有用,将图像从 sRGB 转换为 AdobeRGB1998 配置文件。生成的 TIFF 文件具有正确的 ICC 配置文件并包含完整的 Alpha 通道 ( 258/BitsPerSample: [8, 8, 8, 8], 277/SamplesPerPixels: 4, 34675/ICCProfile: [...]
)。我能看到的唯一小问题是压缩从 LZW 更改为无压缩,DPI 从 300 更改为 72(+ XMP 元数据丢失)。
BufferedImage image = ImageIO.read(new File("C:\\Downloads\\sandal.tif"));
ICC_ColorSpace ics = (ICC_ColorSpace) ColorSpaces.getColorSpace(ColorSpaces.CS_ADOBE_RGB_1998);
ColorConvertOp cco = new ColorConvertOp(ics, null);
BufferedImage result = cco.filter(image, null);
File tempFile = File.createTempFile("test-", ".tif");
System.out.println("tempFile: " + tempFile); // Just to know where to look
ImageIO.write(result, "TIFF", tempFile);
如您所见,这里唯一真正的区别是如何获得 ICC 配置文件/色彩空间。
如果您想保留元数据和/或控制压缩,这也是可能的。下面的代码基本相同(但保留了 LZW 压缩和 300dpi),不幸的是它有点冗长:
try (ImageInputStream input = ImageIO.createImageInputStream(new File("C:\\Downloads\\sandal.tif"))) {
ImageReader reader = ImageIO.getImageReaders(input).next();
reader.setInput(input);
IIOImage imageAndMeta = reader.readAll(0, reader.getDefaultReadParam());
reader.dispose();
ICC_ColorSpace ics = (ICC_ColorSpace) ColorSpaces.getColorSpace(ColorSpaces.CS_ADOBE_RGB_1998);
ColorConvertOp cco = new ColorConvertOp(ics, null);
BufferedImage result = cco.filter((BufferedImage) imageAndMeta.getRenderedImage(), null);
imageAndMeta.setRenderedImage(result);
File tempFile = File.createTempFile("test-", ".tif");
System.err.println("tempFile: " + tempFile);
ImageWriter tiffWriter = ImageIO.getImageWritersByFormatName("TIFF").next();
try (ImageOutputStream stream = ImageIO.createImageOutputStream(tempFile)) {
tiffWriter.setOutput(stream);
ImageWriteParam writeParam = tiffWriter.getDefaultWriteParam();
// If you want a specific compression, uncommment these lines
// The default setting is to copy from metadata
// writeParam.setCompressionMode(ImageWriteParam.MODE_EXPLICIT);
// Allowed compression type values are:
// "None", "CCITT RLE", "CCITT T.4", "CCITT T.6","LZW", "JPEG", "ZLib", "PackBits" and "Deflate"
// writeParam.setCompressionType("PackBits");
tiffWriter.write(null, imageAndMeta, writeParam);
}
tiffWriter.dispose();
}
(由于某种原因,XMP 元数据仍然从输出中删除,我相信这是一个错误)。
编写器当前不支持平铺,但writeParam
将来可能会由其控制(使用标准 API 来实现)。您的原始图像没有平铺,所以我想这不是一个问题。