3

我有一个网络应用程序,用户可以在其中上传图片以创建他们的画廊。几年前,当我编写应用程序时,我选择了ImageMagick,并且我使用 ImageMagick 完成了所有的裁剪和调整大小。

现在我正在从头开始重写应用程序,我将 ImageMagick 替换为原生GDI+操作,但是我对 GDI+ 了解得越多,我就越害怕自己做出了错误的选择。

我到处读到 GDI+ 是用于桌面的,不应该在服务器应用程序上使用。我不知道细节,但我想这是为了内存消耗,实际上我可以看到GDI+ 使用更多的内存来对同一图像执行相同的操作(裁剪和调整大小)而不是 ImageMagick(老实说GDI+ 是更快)。

我相信 GDI+、ImageMagick 或任何其他库对于这些基本操作应该或多或少相同,并且我喜欢使用本机 GDI+ 的想法,相信任何与.NET一起发布的 MS至少应该没问题。

什么是正确的方法/工具?

这是我用来裁剪的代码:

internal Image Crop(Image image, Rectangle r)
{
    Bitmap bmpCrop;
    using (Bitmap bmpImage = new Bitmap(image))
    {
        bmpCrop = bmpImage.Clone(r, bmpImage.PixelFormat);
        bmpImage.Dispose();
    }
    return (Image)(bmpCrop);
}

这是我用来调整大小的代码:

internal Image ResizeTo(Image sourceImage, int width, int height)
{
    System.Drawing.Image newImage = new Bitmap(width, height);
    using (Graphics gr = Graphics.FromImage(newImage))
    {
        gr.SmoothingMode = SmoothingMode.AntiAlias;
        gr.InterpolationMode = InterpolationMode.HighQualityBicubic;
        gr.PixelOffsetMode = PixelOffsetMode.HighQuality;
        gr.DrawImage(sourceImage, new Rectangle(0, 0, width, height));
        gr.Dispose();
    }
    return newImage;
}
4

2 回答 2

2

你能链接到人们说 GDI+ 不应该在服务器上使用的地方吗?也许他们知道一些我不知道的事情。

我知道一些关于 GDI+ 如何工作的事情,但对 ImageMagick 一无所知。我确实发生在描述 ImageMagick 架构的页面上:http ://www.imagemagick.org/script/architecture.php

似乎 ImageMagick 将在内部将图像转换为具有 4 个通道和特定位深度(通常为每个通道 16 位)的未压缩格式,并使用未压缩数据进行工作,这些数据可能在内存中或磁盘上,具体取决于大小。'identify -version' 会告诉你你的位深度是多少。我的印象是,实际上 ImageMagick 通常会在内部使用 64 位 RGBA 缓冲区,除非您使用将使用 32 位 RGBA 的 Q8 版本。它也可以使用多个线程,但我怀疑这会很重要,除非你正在处理非常大的图像。(如果您正在处理非常大的图像,ImageMagick 无疑是赢家。)

GDI+ 位图对象将始终将未压缩的数据存储在内存中,并且通常默认为 32 位 RGBA。那和 32 位 RGB 可能是最有效的格式。GDI+ 是一个绘图库,它不是为大图像设计的,但至少 Bitmap 对象不会保存像素数据和图像元数据的内存以外的任何资源(与流行的看法相反,它们不包含 HBITMAP对象)。

所以他们看起来和我很相似。对于您的用例,我不能说一个明显优于另一个。如果您使用 imagemagick,您可能应该使用 Q8 构建来提高速度和内存,除非额外的精度对您很重要。

看起来如果您唯一的操作是加载、保存、缩放和裁剪,那么您应该能够在以后轻松替换实现。

除非您需要使用元文件,否则您可能应该在内部使用位图对象而不是图像。然后,您不必在 Crop 函数中创建中间位图对象。该中间对象可能是您观察到的一些额外内存消耗的原因。如果您从外部来源获取 Image 对象,我建议您尝试将它们转换为 Bitmap 并在这不起作用时创建一个新的 Bitmap。

此外,“using”语句会自动调用 Dispose,因此无需显式调用它。

于 2012-06-12T02:20:58.917 回答
0

我自己写了一些东西:

public void ResizeImageAndRatio(string origFileLocation, string newFileLocation, string origFileName, string newFileName, int newWidth, int newHeight, bool resizeIfWider)
{
    System.Drawing.Image initImage = System.Drawing.Image.FromFile(origFileLocation + origFileName);
    int templateWidth = newWidth;
    int templateHeight = newHeight;
    double templateRate = double.Parse(templateWidth.ToString()) / templateHeight;
    double initRate = double.Parse(initImage.Width.ToString()) / initImage.Height;
    if (templateRate == initRate)
    {
        System.Drawing.Image templateImage = new System.Drawing.Bitmap(templateWidth, templateHeight);
        System.Drawing.Graphics templateG = System.Drawing.Graphics.FromImage(templateImage);
        templateG.InterpolationMode = System.Drawing.Drawing2D.InterpolationMode.High;
        templateG.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;
        templateG.Clear(Color.White);
        templateG.DrawImage(initImage, new System.Drawing.Rectangle(0, 0, templateWidth, templateHeight), new System.Drawing.Rectangle(0, 0, initImage.Width, initImage.Height), System.Drawing.GraphicsUnit.Pixel);
        templateImage.Save(newFileLocation + newFileName, System.Drawing.Imaging.ImageFormat.Jpeg);
    }
    else
    {
        System.Drawing.Image pickedImage = null;
        System.Drawing.Graphics pickedG = null;

        Rectangle fromR = new Rectangle(0, 0, 0, 0);
        Rectangle toR = new Rectangle(0, 0, 0, 0);

        if (templateRate > initRate)
        {
            pickedImage = new System.Drawing.Bitmap(initImage.Width, int.Parse(Math.Floor(initImage.Width / templateRate).ToString()));
            pickedG = System.Drawing.Graphics.FromImage(pickedImage);

            fromR.X = 0;
            fromR.Y = int.Parse(Math.Floor((initImage.Height - initImage.Width / templateRate) / 2).ToString());
            fromR.Width = initImage.Width;
            fromR.Height = int.Parse(Math.Floor(initImage.Width / templateRate).ToString());

            toR.X = 0;
            toR.Y = 0;
            toR.Width = initImage.Width;
            toR.Height = int.Parse(Math.Floor(initImage.Width / templateRate).ToString());
        }
        else
        {
            pickedImage = new System.Drawing.Bitmap(int.Parse(Math.Floor(initImage.Height * templateRate).ToString()), initImage.Height);
            pickedG = System.Drawing.Graphics.FromImage(pickedImage);

            fromR.X = int.Parse(Math.Floor((initImage.Width - initImage.Height * templateRate) / 2).ToString());
            fromR.Y = 0;
            fromR.Width = int.Parse(Math.Floor(initImage.Height * templateRate).ToString());
            fromR.Height = initImage.Height;

            toR.X = 0;
            toR.Y = 0;
            toR.Width = int.Parse(Math.Floor(initImage.Height * templateRate).ToString());
            toR.Height = initImage.Height;
        }

        pickedG.InterpolationMode = System.Drawing.Drawing2D.InterpolationMode.HighQualityBicubic;
        pickedG.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;

        pickedG.DrawImage(initImage, toR, fromR, System.Drawing.GraphicsUnit.Pixel);

        System.Drawing.Image templateImage = new System.Drawing.Bitmap(templateWidth, templateHeight);
        System.Drawing.Graphics templateG = System.Drawing.Graphics.FromImage(templateImage);
        templateG.InterpolationMode = System.Drawing.Drawing2D.InterpolationMode.High;
        templateG.SmoothingMode = System.Drawing.Drawing2D.SmoothingMode.HighQuality;
        templateG.Clear(Color.White);
        templateG.DrawImage(pickedImage, new System.Drawing.Rectangle(0, 0, templateWidth, templateHeight), new System.Drawing.Rectangle(0, 0, pickedImage.Width, pickedImage.Height), System.Drawing.GraphicsUnit.Pixel);
        templateImage.Save(newFileLocation + newFileName, System.Drawing.Imaging.ImageFormat.Jpeg);

        templateG.Dispose();
        templateImage.Dispose();

        pickedG.Dispose();
        pickedImage.Dispose();
    }
    initImage.Dispose();
}
于 2012-06-10T12:58:40.443 回答