我正在开发一个小型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);
答案 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);
}