绘制图像时:System.Runtime.InteropServices.ExternalException:GDI中发生一般错误

时间:2009-11-20 17:33:12

标签: c# gdi+ winforms

我有一个从Panel创建的全局Graphics对象。定期从磁盘中拾取图像并使用Graphics.DrawImage()将其绘制到面板中。它适用于几次迭代,然后我得到以下有用的例外:

System.Runtime.InteropServices.ExternalException: A generic error occurred in GDI+.
at System.Drawing.Graphics.CheckErrorStatus(Int32 status)
at System.Drawing.Graphics.DrawImage(Image image, Int32 x, Int32 y)
at System.Drawing.Graphics.DrawImage(Image image, Point point)

当我完成图像对象处理时,我排除了内存泄漏。我知道图像没有损坏,并且可以正常读取,因为程序在面板停止显示之前会执行一段时间。

我在使用PictureBox时遇到了同样的问题,但这次至少我得到了一个错误而不是什么。

我检查了任务管理器中的GDI对象和USER对象,但是当应用程序正常工作时,它们总是大约65个用户对象和165个GDI对象。

我确实需要尽快解决这个问题,并不是说我可以在.NET系统库中保留断点并查看确切执行失败的位置。

提前致谢。

编辑:这是显示代码:

private void DrawImage(Image image)
{
  Point leftCorner = new Point((this.Bounds.Width / 2) - (image.Width / 2), (this.Bounds.Height / 2) - (image.Height / 2));
  _graphics.DrawImage(image, leftCorner);
}

图片加载代码:

private void LoadImage(string filename, ref Image image)
{
  MemoryStream memoryStream = DecryptImageBinary(Settings.Default.ImagePath + filename, _cryptPassword);

  image = Image.FromStream(memoryStream);

  memoryStream.Close();
  memoryStream.Dispose();
  memoryStream = null;
}

_image是全局的,其引用在LoadImage中更新。它们作为参数传递,因为我想从尽可能少的地方更改全局引用,并保持其他方法自包含。 _graphics也是全球性的。

我还有一个webBrowser控件用于网站,我要么一次显示图像或网站。当有时间显示图像时,执行以下代码:

webBrowser.Visible = false;
panel.Visible = true;
DrawImage(_image)
_image.Dispose();
_image = null;

_image正在引用预先加载的图像。

希望这有帮助。

2 个答案:

答案 0 :(得分:15)

你的问题与我的想法相似,但并不完全。加载图像时,您将从MemoryStream加载它。您必须在图片的生命周期内保持流打开,请参阅MSDN Image.FromStream

  

您必须在图像的生命周期内保持流打开。

解决方案是在FromImage函数中复制您的图像:

private void LoadImage(string filename, ref Image image)
{
  using (MemoryStream memoryStream = DecryptImageBinary(Settings.Default.ImagePath + filename, _cryptPassword))
  {
      using (tmpImage = Image.FromStream(memoryStream))
      { 
         image = new Bitmap(tmpImage);
      }
  }

}

与我提到的处理问题类似,当底层流被垃圾收集时,图像似乎会起作用然后随机失败。

答案 1 :(得分:2)

如果没有更多的代码,那么这里没有足够的正确诊断,但是,有一点需要注意的是,您可能已经在某个时刻将图像放置在图像上并且仅在垃圾收集器运行之后您的代码失败了。您是否在任何地方使用克隆图像?令我惊讶的一件事是,如果你做一个直的clone图像,你不是克隆图像所依赖的底层位图,只是图像结构,来创建一个图像的正确副本你必须创建一个新的图像:

var newImage = new Bitmap(img)

作为

var newImage = oldImg.Clone();
oldImg.Dispose();
...
gr.DrawImage(newImage, new Rectangle(0,0,newImage.Width,newImage.Height);

会工作一段时间,但随后会失败......