12

通过我对MaxTo的自动崩溃收集,我得到了以下崩溃报告:

V8.12.0.0 - System.ComponentModel.Win32Exception - :Void UpdateLayered():0
Version: MaxTo8.12.0.0
Exception: System.ComponentModel.Win32Exception
Error message: Not enough storage is available to process this command
Stack trace: 
  at System.Windows.Forms.Form.UpdateLayered()
  at System.Windows.Forms.Form.OnHandleCreated(EventArgs e)
  at System.Windows.Forms.Control.WmCreate(Message& m)
  at System.Windows.Forms.Control.WndProc(Message& m)
  at System.Windows.Forms.ScrollableControl.WndProc(Message& m)
  at System.Windows.Forms.ContainerControl.WndProc(Message& m)
  at System.Windows.Forms.Form.WmCreate(Message& m)
  at System.Windows.Forms.Form.WndProc(Message& m)
  at MaxTo.MainForm.WndProc(Message& m)
  at System.Windows.Forms.Control.ControlNativeWindow.OnMessage(Message& m)
  at System.Windows.Forms.Control.ControlNativeWindow.WndProc(Message& m)
  at System.Windows.Forms.NativeWindow.Callback(IntPtr hWnd, Int32 msg, IntPtr wparam, IntPtr lparam)

另一个堆栈跟踪:

Version: MaxTo2009.9.0.0
Exception: System.ComponentModel.Win32Exception
Error message: Not enough storage is available to process this command
Stack trace: 
  at System.Windows.Forms.Form.UpdateLayered()
  at System.Windows.Forms.Form.OnHandleCreated(EventArgs e)
  at System.Windows.Forms.Control.WmCreate(Message& m)
  at System.Windows.Forms.Control.WndProc(Message& m)
  at System.Windows.Forms.ScrollableControl.WndProc(Message& m)
  at System.Windows.Forms.ContainerControl.WndProc(Message& m)
  at System.Windows.Forms.Form.WmCreate(Message& m)
  at System.Windows.Forms.Form.WndProc(Message& m)
  at System.Windows.Forms.Control.ControlNativeWindow.OnMessage(Message& m)
  at System.Windows.Forms.Control.ControlNativeWindow.WndProc(Message& m)
  at System.Windows.Forms.NativeWindow.Callback(IntPtr hWnd, Int32 msg, IntPtr wparam, IntPtr lparam)

在这个最新的堆栈跟踪中,根本没有提到 MaxTo,而且我遇到的 90% 的崩溃都与上述类似的堆栈跟踪有关。

在网上阅读我发现如果您忘记释放或处置变量,此错误很常见。当查看我的WndProc时,似乎有时会出现问题,我找不到一个地方可以挂起对任何对象的引用。除了一个变量之外的所有变量都是 WndProc 的本地变量,因此在方法终止时应该进行垃圾回收。

protected override void WndProc(ref Message m)
{
    base.WndProc(ref m); // I'm assuming the first trace can be caught here
    IntPtr hwnd = m.WParam;
    // Our hook tells us something got maximized
    if (Win32Import.UWM_MAXIMIZE == (UInt32)m.Msg)
    {
        // Figure out if we are temporarily disabled or using alternative profiles
        KeyStateInfo keyState = KeyboardInfo.GetKeyState(Settings.AlternativeProfileKey);
        Rectangle r = FindRectangle(MousePosition, (Settings.EnableAlternativeProfile && keyState.IsPressed ? AlternativeRegions : Regions));
        // Did we find a rectangle to place it in?
        if (r != Rectangle.Empty)
        {
            Rectangle position = Win32Import.GetWindowRectangle(hwnd);
            Rectangle previousPos = GetLocation(hwnd);
            if (position == r && previousPos != Rectangle.Empty)
            {
                // We are restoring the original position
                Win32Import.SetWindowPos(hwnd, IntPtr.Zero, previousPos.X, previousPos.Y, previousPos.Width, previousPos.Height, Win32Import.SWP_NOZORDER | Win32Import.SWP_NOSENDCHANGING);
            }
            else
            {
                // We are maximizing to a region
                Win32Import.ShowWindow(hwnd, Win32Import.WindowShowStyle.Restore);
                Win32Import.SetWindowPos(hwnd, IntPtr.Zero, r.X, r.Y, r.Width, r.Height, Win32Import.SWP_NOZORDER | Win32Import.SWP_NOSENDCHANGING);
                // Make sure we remember this location
                RememberLocation(hwnd, position);
            }
        }
    }
    else if (MaxTo64WindowHandleMessage == m.Msg)
    {
        // Store the window handle of our 64-bit subprocess
        SubProcess64WindowHandle = m.WParam;
    }
}

即使在多天运行程序时,我也无法重现该错误。

我的假设是系统的未碎片化内存或 GDI 句柄都很低,但我无法在任何地方确认这一点。似乎没有关于此错误的任何好的文档。

任何想法它可能是什么?我能做些什么来防止这个错误吗?

更新:由于缺乏一个像样的解决方案,这个问题被重新打开了更多的堆栈跟踪。简单地忽略它并不能解决问题。

4

4 回答 4

12

泄漏或使用到许多 GDI 对象/句柄。这些可能会导致资源堆短缺。您可能无法重现,因为您的用户可能正在运行其他 GDI 资源繁重的程序或使用终端服务器,在这种情况下,他们必须与其他用户共享一些堆。请参阅系统错误。代码:8. 没有足够的存储空间来处理这个命令

在这里,您可以阅读有关用于诊断桌面堆问题的桌面堆监视器工具。

这里这里这里是 GDI 泄漏检测工具。

于 2009-02-14T12:15:58.413 回答
6

您的程序可能正在泄漏内核资源。开始使用 Taskmgr.exe 诊断此问题。查看 + 选择列,检查用户对象、GDI 对象和句柄计数。运行您的程序并观察其中是否有任何稳定增加。一旦其中一个达到 10,000,您的程序就会死掉。

通过一种快速查看实际泄漏的方法,您可以开始注释代码以查看泄漏发生的位置。它可能与您的“钩子”有关。

于 2009-02-14T13:02:29.360 回答
1

我有许多具有自己资源的自定义 Windows 控件,因此当我创建许多控件时会出现此错误。为了解决这个问题,我在我的库中创建了资源文件,并在我的组件代码中使用了外部资源而不是资源。之后我的异常消失了,已经用 3 倍以上的打开表单进行了测试,并且这个错误消失了。所以看起来这是一个解决方案。

于 2013-07-16T15:04:45.997 回答
1

问题可能不在于您的 WndProc - 您在调用堆栈中看到它的原因是因为 Windows 上与 GUI 相关的几乎所有内容都通过 WIN32 窗口过程。在您的控件中覆盖它只是为您提供了一个挂钩点,以便在完成更高级别的 .NET 框架处理之前处理低级别的内容。

这将是一个完整的黑暗中的镜头,但也许这篇文章可能是相关的?- 不过,可能不是那些堆栈跟踪。

于 2010-09-21T07:46:42.463 回答