C#图像大小调整 - 丢失EXIF

时间:2013-07-23 15:20:58

标签: c# image image-processing metadata gdi

是的...我已经看到了与此问题相关的其他帖子,是的...我已经用Google搜索过了。

但到目前为止,我无法得到我需要的结果。

我正在加载以300 dpi拍摄的大图像,我需要调整它的大小。

我知道......我知道...... dpi是相对的,并不重要......重要的是像素尺寸:

  

DPI基本上是打印图像时对应于英寸的像素数,而不是在屏幕上查看时的像素数。因此,通过增加图像的DPI,您不会增加屏幕上图像的大小。您只能提高打印质量。

即使存储在图像EXIF中的DPI信息有些无用,也会给我带来麻烦。

我正在调整大小的图像是丢失原始exif 信息,包括水平和垂直分辨率(dpi),因此它以96 dpi的默认值保存。可能的原因是只有JPEG和其他格式才能保存元数据信息。

最终图像结果应如下所示:275x375 at 300dpi 而是看起来像这样:275x375 at 96dpi

你可以说他们是一样的,我同意,但我们有一个用于加载这些图像的corel绘图脚本,并且由于这个dpi信息不同,它会在文档中放置不同的大小。

以下是我用于调整大小的内容:

public System.Drawing.Bitmap ResizeImage(System.Drawing.Image image, int width, int height)
    {
        Bitmap result = new Bitmap(width, height);

        // set the resolutions the same to avoid cropping due to resolution differences
        result.SetResolution(image.HorizontalResolution, image.VerticalResolution);

        //use a graphics object to draw the resized image into the bitmap
        using (Graphics graphics = Graphics.FromImage(result))
        {
            //set the resize quality modes to high quality
            graphics.CompositingQuality = System.Drawing.Drawing2D.CompositingQuality.HighQuality;
            graphics.InterpolationMode = System.Drawing.Drawing2D.InterpolationMode.High;
            graphics.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;

            //draw the image into the target bitmap
            graphics.DrawImage(image, 0, 0, result.Width, result.Height);
        }

        //return the resulting bitmap
        return result;
    }

这项工作做得很好,但丢失了EXIF信息。

将SetResolution设置为SetResolution(300,300)不起作用!

我看了一下阅读并更改了图片的EXIF信息,我试过了:

public void setImageDpi(string Filename, string NewRes)
    {
        Image Pic;
        PropertyItem[] PropertyItems;
        byte[] bDescription = new Byte[NewRes.Length];
        int i;
        string FilenameTemp;
        System.Drawing.Imaging.Encoder Enc = System.Drawing.Imaging.Encoder.Transformation;
        EncoderParameters EncParms = new EncoderParameters(1);
        EncoderParameter EncParm;
        ImageCodecInfo CodecInfo = GetEncoderInfo("image/jpeg");

        // copy description into byte array
        for (i = 0; i < NewRes.Length; i++) bDescription[i] = (byte)NewRes[i];

        // load the image to change
        Pic = Image.FromFile(Filename);

        foreach (PropertyItem item in Pic.PropertyItems)
        {
            if (item.Id == 282 || item.Id == 283)
            {
                PropertyItem myProperty = item;
                myProperty.Value = bDescription;
                myProperty.Type = 2;
                myProperty.Len = NewRes.Length;
                Pic.SetPropertyItem(item);
                Console.WriteLine(item.Type);
            }
        }

        // we cannot store in the same image, so use a temporary image instead
        FilenameTemp = Filename + ".temp";

        // for lossless rewriting must rotate the image by 90 degrees!
        EncParm = new EncoderParameter(Enc, (long)EncoderValue.TransformRotate90);
        EncParms.Param[0] = EncParm;

        // now write the rotated image with new description
        Pic.Save(FilenameTemp, CodecInfo, EncParms);

        // for computers with low memory and large pictures: release memory now
        Pic.Dispose();
        Pic = null;
        GC.Collect();

        // delete the original file, will be replaced later
        System.IO.File.Delete(Filename);

        // now must rotate back the written picture
        Pic = Image.FromFile(FilenameTemp);
        EncParm = new EncoderParameter(Enc, (long)EncoderValue.TransformRotate270);
        EncParms.Param[0] = EncParm;
        Pic.Save(Filename, CodecInfo, EncParms);

        // release memory now
        Pic.Dispose();
        Pic = null;
        GC.Collect();

        // delete the temporary picture
        System.IO.File.Delete(FilenameTemp);
    }

这也不起作用。

我尝试在此过程的后期查找并更改DPI(282和283)的EXIF信息:

        Encoding _Encoding = Encoding.UTF8;
        Image theImage = Image.FromFile("somepath");

        PropertyItem propItem282 = theImage.GetPropertyItem(282);
        propItem282.Value = _Encoding.GetBytes("300" + '\0');
        theImage.SetPropertyItem(propItem282);

        PropertyItem propItem283 = theImage.GetPropertyItem(283);
        propItem283.Value = _Encoding.GetBytes("300" + '\0');
        theImage.SetPropertyItem(propItem283);

        theImage.Save("somepath");

但程序崩溃说找不到属性。

如果该属性不存在,显然我无法添加它:

  

PropertyItem不能用作独立对象。 PropertyItem对象旨在由派生自Image的类使用。 PropertyItem对象用于检索和更改现有图像文件的元数据,而不是创建元数据。因此,PropertyItem类没有已定义的Public构造函数,也无法创建PropertyItem对象的实例。

我被卡住了...我需要的是一张dpi设置为300的已调整大小的图片,它应该不会那么难。

任何帮助非常感谢。感谢

1 个答案:

答案 0 :(得分:14)

以下代码对我有用:

const string InputFileName = "test_input.jpg";
const string OutputFileName = "test_output.jpg";
var newSize = new Size(640, 480);

using (var bmpInput = Image.FromFile(InputFileName))
{
    using (var bmpOutput = new Bitmap(bmpInput, newSize))
    {
        foreach (var id in bmpInput.PropertyIdList)
            bmpOutput.SetPropertyItem(bmpInput.GetPropertyItem(id));
        bmpOutput.SetResolution(300.0f, 300.0f);
        bmpOutput.Save(OutputFileName, ImageFormat.Jpeg);
    }
}

当我检查输出文件时,我可以看到EXIF数据,并且DPI已更改为300。