在C#中加载32位灰度TIFF图像

时间:2015-03-20 10:10:13

标签: c# image

我正在开发一个小型C#工具,必须能够加载TIFF图像,将图像裁剪为一定大小,并将其另存为PNG文件。

我有大约28000x256像素和32位位深度的大灰度TIFF图像。当我尝试用我的工具处理图像时,它只输出一个空白的白色图像。

此外,当我尝试使用Windows照片查看器打开原始TIFF图像(不是使用我的工具处理的图像)时,它还会显示空白图像。一些其他应用,例如ImageJ,正确显示图像。这有什么问题?

我加载图片的代码如下:

Image image = Bitmap.FromFile(path.LocalPath);
int width = image.Width;
int height = image.Height;

Bitmap bmp = new Bitmap(width, height);
Graphics g = Graphics.FromImage(bmp);

1 个答案:

答案 0 :(得分:0)

问题是C#(或更好的说底层API)无法处理Colordepth大于8bit的灰度图像。

我建议使用LibTiff.NET来处理TIFF图像。 当我遇到这样的问题时,我将TIFF图像原始数据加载到数组中

using (var inputImage = Tiff.Open(image, "r"))
{
    width = inputImage.GetField(TiffTag.IMAGEWIDTH)[0].ToInt();
    height = inputImage.GetField(TiffTag.IMAGELENGTH)[0].ToInt();
    inputImageData = new byte[width * height * bytePerPixel];
    var offset = 0;
    for (int i = 0; i < inputImage.NumberOfStrips(); i++)
    {
        offset += inputImage.ReadRawStrip(i, inputImageData, offset, (int)inputImage.RawStripSize(i));
    }
}

然后必须将字节转换为uint的数组(在我的例子中,imagedata只有16位,所以我使用ushort)记住要处理数据的字节顺序!< / p>

// has to be done by hand to ensure endiannes is kept correctly.
var outputImageData = new ushort[inputImageData.Length / 2];

for (var i = 0; i < outputImageData.Length; i++)
{
    outputImageData[i] = (ushort)((inputImageData[i * 2 + 1]) + (ushort)(inputImageData[i * 2] << 8));
}

然后,您可以使用常规阵列操作来操作图像。我建议你使用普通的数组操作而不是Lambda-Expressions,因为它们要快得多。 (在My Scenario 100s vs 2s Runtime中)

最后,您可以再次使用LibTiff保存图像

using (var output = Tiff.Open(imageout, "w"))
{
    output.SetField(TiffTag.IMAGEWIDTH, width);
    output.SetField(TiffTag.IMAGELENGTH, height);
    output.SetField(TiffTag.SAMPLESPERPIXEL, 1);
    output.SetField(TiffTag.BITSPERSAMPLE, 16);
    output.SetField(TiffTag.ROWSPERSTRIP, height);
    output.SetField(TiffTag.PHOTOMETRIC, Photometric.MINISBLACK);
    output.SetField(TiffTag.FILLORDER, FillOrder.MSB2LSB);

    // Transform to Byte-Array
    var buffer = new byte[outputImageData.Length * sizeof(ushort)];
    Buffer.BlockCopy(outputImageData, 0, buffer, 0, buffer.Length);

    // Write it to Image
    output.WriteRawStrip(0, buffer, buffer.Length);
}