.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:tempimgoriginal_img.jpg";
string newFile = @"D:tempimgnew_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);
        }
    }

最新更新