如果我执行以下代码,则OutOfMemoryException会出现在任一行

using (Bitmap bitmap1 = new Bitmap(FrameToFilePath(interval.Start - 1)))

或线
using (Bitmap bitmap2 = new Bitmap(FrameToFilePath(interval.End + 1)))

当内部的for语句执行大约1000次时。

但是,我不知道为什么发生OutOfMemoryException。我想我已经写了足够的using来处理Bitmap对象。内存泄漏发生在哪里?
class Program
{
    static void Main(string[] args)
    {
        // Some code to initialize List<Interval> intervals

        Parallel.ForEach(intervals, interval =>
        {
            using (Bitmap bitmap1 = new Bitmap(FrameToFilePath(interval.Start - 1)))
            using (Bitmap bitmap2 = new Bitmap(FrameToFilePath(interval.End + 1)))
            {
                for (int i = interval.Start; i <= interval.End; i++)
                {
                    ColorMatrix colorMatrix = new ColorMatrix(); // Identity matrix
                    colorMatrix.Matrix33 = (i - interval.Start + 1F) / (interval.Span + 1F); // Alpha

                    using (ImageAttributes imageAttributes = new ImageAttributes())
                    using (Bitmap intermediate = new Bitmap(bitmap1.Width, bitmap1.Height, PixelFormat.Format32bppArgb))
                    using (Graphics graphics = Graphics.FromImage(intermediate))
                    {
                        imageAttributes.SetColorMatrix(colorMatrix, ColorMatrixFlag.Default, ColorAdjustType.Bitmap);

                        graphics.CompositingMode = System.Drawing.Drawing2D.CompositingMode.SourceOver;

                        graphics.DrawImage(bitmap1, 0, 0);
                        graphics.DrawImage(bitmap2, new Rectangle(0, 0, intermediate.Width, intermediate.Height), 0, 0, bitmap2.Width, bitmap2.Height, GraphicsUnit.Pixel, imageAttributes);

                        intermediate.Save(FrameToFilePath(i), ImageFormat.Png);
                    }
                }
            }
        });
    }

    static string FrameToFilePath(int frame)
    {
        return string.Format(@"C:\(some path)\frames\frame-{0:00000}.png", frame);
    }
}

class Interval
{
    public int Start { get; set; }
    public int End { get; set; }
    public int Span { get { return End - Start + 1; } }
}

编辑:好的。可能是因为Parallel.ForEach在其他任务结束之前产生了任务,所以启动了越来越多的任务,但是Bitmap未被GC,因为这些任务没有结束。好吧,如果是这种情况,如何解决它以避免OutofMemoryException?我不确定是否是这种情况...

最佳答案

我遇到了类似的issue myself(但不足以让我想将其标记为重复)ojit_a。您可以做的是将ParallelOptions.MaximumDegreeOfParallelism设置为在发现工作停顿时不创建额外的任务。

Parallel.ForEach(intervals,
    new ParallelOptions { MaxDegreeOfParallelism = Environment.ProcessorCount },
    (interval) =>
        {
           //...
        });

出现此问题的原因是,绑定(bind)的对象不是CPU,而是读取FrameToFilePath返回的文件期间驱动器的IO。这会导致CPU%较低,并且Parallel.ForEach会检测到该%CPU,并尝试生成更多任务以使使用率达到100%。设置最大并行度后,一旦创建了一定数量的任务,它便停止尝试达到100%。您可以使用大于Environment.ProcessorCount的数字,但主要瓶颈是文件IO。

10-04 18:42