使用有损jpeg压缩多页tiff图像

时间:2018-05-27 17:32:23

标签: java image compression tiff

我需要压缩一个有几个灰色16位图像(多页)的tif文件。我尝试使用ImageIO,如下所示:Tiff compression using Java ImageIO最初,将在tif文件中的每个图像来自另一个tiff文件。当我想使用压缩器时,我有以下选择:

  1. CCITT RLE,CCITT T.4,CCITT T.6:他们给我错误:“javax.imageio.IIOException:I / O错误写入TIFF文件!”
  2. LZW。我不能用它。我的图像是16位,LZW增加了16位图像的大小
  3. JPEG。不适用于16位图像。
  4. ZLIB。即使我指定setCompressionQuality(0.0f);
  5. ,它也只减少10%
  6. 紧缩位。不压缩。
  7. 放气。像ZLIB一样。
  8. EXIF JPEG。它给了我错误:“javax.imageio.IIOException:不支持旧JPEG压缩!”
  9. 有没有其他选择吗?我看到了一个apache映像库,但tif压缩只支持上面或更少的选项。有谁知道JPEG2000压缩器?还有其他任何选择吗?

2 个答案:

答案 0 :(得分:1)

PNG无损压缩16位图像。图书馆和公用事业广泛可用。 JPEG2000有一个有损16位模式,但你必须找到一些支持它的软件。 Open JPEG可能。

但是我不得不问:当你有可接受的图像质量时,你的标准是什么?如果它是可视的,那么你最终可能会以正常的JPEG结果,每像素的有效位小于8位。

答案 1 :(得分:0)

将图像16位缩小为8位。假设您有一个byte []变量plane16,其中包含图像的所有像素。

注意:我的byte [] plane16从16位图像中获取数据,但字节为8位= 1byte。因此,该数组的行中的2个元素是2byte = 16bit。这就是我在操作前将其转换为short []的原因。如果你从short []开始,请省略“ByteBuffer.wrap(plane16).order(ByteOrder.LITTLE_ENDIAN).asShortBuffer()。get(shorts);”

byte[] plane16; //Fill it with your image!!!

//Do stuff with imageIO. Set writer and compresion method
ImageIO.scanForPlugins();
TIFFImageWriterSpi tiffspi = new TIFFImageWriterSpi();
javax.imageio.ImageWriter writerIO = tiffspi.createWriterInstance();
ImageWriteParam param = writerIO.getDefaultWriteParam();
param.setCompressionMode(ImageWriteParam.MODE_EXPLICIT);
param.setCompressionType("ZLib");
param.setCompressionQuality(0.5f);
File fOutputFile = new File(route+".tif");
ImageOutputStream ios = ImageIO.createImageOutputStream(fOutputFile);
writerIO.setOutput(ios);

//Reducing 16bit to 8bit
short[] shorts = new short[plane16.length/2];
ByteBuffer.wrap(plane16).order(ByteOrder.LITTLE_ENDIAN).asShortBuffer().get(shorts);

int max = 0;
int min = 999999;
for (int v = 0; v<shorts.length;v++){
    if (max<shorts[v]) max = shorts[v];
    if (min>shorts[v]) min = shorts[v];
}

double range = 255./(max-min);
byte[] plane8 = new byte[shorts.length];
for (int v = 0; v<plane8.length;v++){
    plane8[v] = (byte) ((shorts[v]+min)*range - 128);   
}                       
//16bit:    
/*BufferedImage convertedGrayscale = new BufferedImage(width,
                            heigth, BufferedImage.TYPE_USHORT_GRAY);    
  convertedGrayscale.getRaster().setDataElements(0, 0, width,
                            heigth, shorts);*/

//8bit:
BufferedImage convertedGrayscale = new BufferedImage(width,
                            heigth, BufferedImage.TYPE_BYTE_GRAY);  
convertedGrayscale.getRaster().setDataElements(0, 0, width,
                            heigth, plane8);

//Save image
//If you have a stack of images in tiff, do this trick. "image" is the image number you are setting inside the tiff. If you only have 1 image, remove the if and take the expression from the else.
if (image!=0){
    writerIO.writeInsert(image, new IIOImage(convertedGrayscale, null, null), param);                   
}else{
    writerIO.write(null, new IIOImage(convertedGrayscale, null, null), param);
}

//do the next only after the last image to be saved
writerIO.dispose();
ios.flush();
ios.close();