.NET - Bitmap.Save 忽略 Windows 7 上的 Bitmap.SetResolution
本文关键字:Bitmap 上的 Windows SetResolution 忽略 Save NET | 更新日期: 2023-09-27 18:34:00
我正在编写一个 .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?forum=netfxbcl
但那里的问题似乎仍未解决。真的没有办法让它工作吗?我必须为此使用额外的库吗?
我找到了可以完成这项工作的解决方法。它不优雅,但...
不要将分辨率应用于原始图像,而是制作它的副本并处理副本:
Bitmap bitmap = (Bitmap)Image.FromFile(originalFile);
Bitmap newBitmap = new Bitmap(bitmap)
newBitmap.SetResolution(200, 200);
newBitmap.Save(newFile, ImageFormat.Jpeg);
现在它可以在Windows 7上运行。去图。
我喜欢Hans Passant的想法,不过,它更干净。我不知道我所做的是否弄乱了图像,是否有重新压缩。
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);
}
}