调整许多位图大小时出现内存不足异常

Out of memory exception when resizing many bitmaps

我需要调整 5000 张图片的大小并将它们保存在单独的文件夹中。我有这样的代码来创建图片的调整大小副本(在 Internet 上找到):

Bitmap ResizeImage(System.Drawing.Image image, int width, int height)
       
 {
            var destRect = new System.Drawing.Rectangle(0, 0, width, height);
            var destImage = new Bitmap(width, height);

            destImage.SetResolution(image.HorizontalResolution, image.VerticalResolution);

            using (var graphics = Graphics.FromImage(destImage))
            {
                graphics.CompositingMode = CompositingMode.SourceCopy;
                graphics.CompositingQuality = CompositingQuality.HighQuality;
                graphics.InterpolationMode = InterpolationMode.HighQualityBicubic;
                graphics.SmoothingMode = SmoothingMode.HighQuality;
                graphics.PixelOffsetMode = PixelOffsetMode.HighQuality;

                using (var wrapMode = new ImageAttributes())
                {
                    wrapMode.SetWrapMode(WrapMode.TileFlipXY);
                    graphics.DrawImage(image, destRect, 0, 0, image.Width, image.Height, GraphicsUnit.Pixel, wrapMode);
                }
            }
            
            return destImage;
        }

我有这段代码,它首先创建所有必要的目录来保存图片,然后开始更改所有图像并将它们保存到新文件夹中。在执行程序的过程中,我的内存开始填充,直到达到 ~ 590mb (~ 1800 images) 的标记,然后抛出 OutOfMemoryException 错误。

void ResizeAllImages()
        {
            var files = Directory.GetFiles(ImagesDirectory, "*", SearchOption.AllDirectories);

            if (!Directory.Exists(ResizedImagesDirectory)) Directory.CreateDirectory(ResizedImagesDirectory);

            var dirs = Directory.GetDirectories(ImagesDirectory);
 
            foreach(var d in dirs) Directory.CreateDirectory(System.IO.Path.Combine(ResizedImagesDirectory, System.IO.Path.GetFileName(d)));

            foreach(var f in files)
            {
                using(var image = ResizeImage(System.Drawing.Image.FromFile(f), 224, 224))
                {
                    var imgName = System.IO.Path.GetFileName(f);

                    var dirName = System.IO.Path.GetFileName(System.IO.Path.GetDirectoryName(f));

                    image.Save(System.IO.Path.Combine(ResizedImagesDirectory, dirName, imgName), ImageFormat.Png);
                }
            }
        }

为了解决这个问题,我尝试在foreach中加入如下两行代码:

GC.Collect();
GC.WaitForPendingFinalizers();

探查器开始显示,在整个程序运行期间,该过程占用约 50mb,但是,当达到之前的约 1800 张图像标记时,我再次遇到 OutOfMemoryException。我怎么解决这个问题。提前谢谢你

根据 MSDN,OutOfMemoryException 可以在某些 non-intuitive 情况下抛出。比如可以在Bitmap.Clone方法中抛出:

OutOfMemoryException: rect is outside of the source bitmap bounds.

我们可以看出您没有使用Clone方法,而是使用了Rect和Bitmap,并且由于您程序的内存占用不超过50MB,因此可以推断问题类似。

要真正解决这个问题,您应该首先隔离有问题的图像,准确验证发生了什么,然后进行相应的处理。