.NET – Bitmap.Save在Windows 7上忽略Bitmap.SetResolution

我正在编写一个.NET 4应用程序,用于导入和保存图像以进行打印。 保存的图像分辨率(DPI不是像素尺寸)必须设置为我们指定的值,以便正确打印。

我们导入的一些图像没有分辨率值(生成它们时出现错误的EXIF),因此我们必须在编写之前纠正它们。 我们使用Bitmap.SetResolution。 它在XP和Windows 8上工作正常,但是当我们在Windows 7上编写(Bitmap.Save)图像时,它们总是用原始分辨率元信息编写,忽略SetResolution。

这是我们做的测试,适用于XP和8,而不是7。

string originalFile = @"D:\temp\img\original_img.jpg"; string newFile = @"D:\temp\img\new_img.jpg"; Bitmap bitmap = (Bitmap)Image.FromFile(originalFile); bitmap.SetResolution(200, 200); bitmap.Save(newFile, ImageFormat.Jpeg); Image image = Image.FromFile(newFile); int dpiX = (int)Math.Round(image.HorizontalResolution, MidpointRounding.ToEven); int dpiY = (int)Math.Round(image.VerticalResolution, MidpointRounding.ToEven); Console.WriteLine("DPI is {0} x {1}", dpiX, dpiY); 

在保存之前,调试始终显示由SetResolution指定的正确分辨率,保存的图像是问题所在。

这可能是这里报道的: http : //social.msdn.microsoft.com/Forums/vstudio/en-US/62368caa-05f4-4798-9c59-5d82f881a97c/systemdrawingbitmapsetresolution-is-completely-broken-on-windows- 7?论坛= netfxbcl

但那里的问题似乎仍未得到解决。 真的没有办法让它发挥作用吗? 我必须为此使用额外的库吗?

Hmya,这是Windows组件中的一个错误。 Windows组总是非常不愿意修复这样的错误,突破性的更改被推迟到下一个Windows版本。 它确实在Windows 8中得到修复。请考虑您正在做的事情有多不寻常,图像的DPI应始终由记录图像的设备设置。 像相机或扫描仪一样,它们永远不会出错。 周围没有任何设备具有200点/英寸的分辨率。

如果您非常渴望找到解决方法,那么您可以考虑修补文件本身。 对于JPEG文件来说并不难,文件头中的字段很容易找到:

 using System.IO; ... public static void SetJpegResolution(string path, int dpi) { using (var jpg = new FileStream(path, FileMode.Open, FileAccess.ReadWrite, FileShare.None)) using (var br = new BinaryReader(jpg)) { bool ok = br.ReadUInt16() == 0xd8ff; // Check header ok = ok && br.ReadUInt16() == 0xe0ff; br.ReadInt16(); // Skip length ok = ok && br.ReadUInt32() == 0x4649464a; // Should be JFIF ok = ok && br.ReadByte() == 0; ok = ok && br.ReadByte() == 0x01; // Major version should be 1 br.ReadByte(); // Skip minor version byte density = br.ReadByte(); ok = ok && (density == 1 || density == 2); if (!ok) throw new Exception("Not a valid JPEG file"); if (density == 2) dpi = (int)Math.Round(dpi / 2.56); var bigendian = BitConverter.GetBytes((short)dpi); Array.Reverse(bigendian); jpg.Write(bigendian, 0, 2); jpg.Write(bigendian, 0, 2); } } 

我找到了可以完成这项工作的解决方法。 它不优雅,但……

而不是将分辨率应用于原始图像,制作它的副本并处理副本:

 Bitmap bitmap = (Bitmap)Image.FromFile(originalFile); Bitmap newBitmap = new Bitmap(bitmap) newBitmap.SetResolution(200, 200); newBitmap.Save(newFile, ImageFormat.Jpeg); 

现在它适用于Windows 7. Go figure。

我喜欢Hans Passant的想法,但它更清洁。 如果有再压缩,我不知道我做的是否与图像混淆。