7

我正在编写一个 .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

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

4

2 回答 2

19

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

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

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

现在它可以在 Windows 7 上运行。去看看。

不过,我喜欢 Hans Passant 的想法,它更简洁。我不知道我所做的是否弄乱了图像,是否有重新压缩。

于 2013-10-07T18:38:26.540 回答
6

嗯,这是 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);
        }
    }
于 2013-10-04T17:38:58.193 回答