4

当我为 GetWindowText 运行下面的代码时,我收到以下错误作为内部异常抛出:

{“试图读取或写入受保护的内存。这通常表明其他内存已损坏。”}

    [DllImport("user32.dll", EntryPoint = "GetWindowTextLength", SetLastError = true)]
    internal static extern int GetWindowTextLength(IntPtr hwnd);

    [DllImport("user32.dll", EntryPoint = "GetWindowText", SetLastError = true)]
    internal static extern int GetWindowText(IntPtr hwnd, ref StringBuilder wndTxt, int MaxCount);

try{
      int strLength = NativeMethods.GetWindowTextLength(wndHandle);
      var wndStr = new StringBuilder(strLength);
      GetWindowText(wndHandle, ref wndStr, wndStr.Capacity);
   }
    catch(Exception e){ LogError(e) }

我有两个问题:

  1. 为什么错误没有被 try catch 捕获?

  2. 知道如何在程序遇到此类错误时阻止程序崩溃,而不是使用 try/catch

干杯

4

2 回答 2

9

1.

有一些异常是无法捕获的。一种类型是 StackOverflow 或 OutOfMemory,因为实际上没有内存可以分配给处理程序运行。另一种类型是通过 Windows 操作系统交付给 CLR。这种机制称为结构化异常处理。这类异常可能非常糟糕,因为 CLR 无法确定其自身的内部状态是否一致,有时称为损坏状态异常。在 .Net 4 中,托管代码默认不处理这些异常。

上述消息来自 AccessViolationException,这是一种损坏状态异常。发生这种情况是因为您正在调用一个写入缓冲区末尾的非托管方法。请参阅这篇关于可能处理这些异常的文章。

2.

这里的示例代码有效吗?您需要确保非托管代码不会写入超过StringBuilder' 缓冲区的末尾。

public static string GetText(IntPtr hWnd)
{
    // Allocate correct string length first
    int length       = GetWindowTextLength(hWnd);
    StringBuilder sb = new StringBuilder(length + 1);
    GetWindowText(hWnd, sb, sb.Capacity);
    return sb.ToString();
}
于 2012-06-01T16:02:59.073 回答
2

由于您提供给 GetWindowText 的参数,调用这些外部方法可能会导致问题。我认为您应该尝试以下方法:

try{
    int strLength = NativeMethods.GetWindowTextLength(wndHandle);
    var wndStr = new StringBuilder(strLength + 1);
    GetWindowText(wndHandle, wndStr, wndStr.Capacity);
   }
catch(Exception e){ LogError(e) }
于 2012-06-01T15:58:49.733 回答