如果我执行以下代码,则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。