1

我有一个 .NET Windows 窗体应用程序,它偶尔会进入挂起状态。当我转储进程并通过 WinDbg/SOS 打开它时,它显示一个线程有一个调用,kernel32!WaitForMultipleObjectsEx尽管我没有在该特定方法中的任何事件对象上调用任何 WaitXXX API。这在我正在检查 InvokedRequired 和 IsDisposed API 中说过。对 InvokedRequired 或 IsDisposed API 的任何调用是否会在内部调用任何 WaitXXX API?

这是线程的调用堆栈。

[HelperMethodFrame_1OBJ: 0e4eedc0] System.Threading.WaitHandle.WaitOneNative(Microsoft.Win32.SafeHandles.SafeWaitHandle, UInt32, Boolean, Boolean)  
0e4eee6c 67b7689f System.Threading.WaitHandle.WaitOne(Int64, Boolean)  
0e4eee88 67b76855 System.Threading.WaitHandle.WaitOne(Int32, Boolean)  
0e4eee9c 65c21a4f System.Windows.Forms.Control.WaitForWaitHandle(System.Threading.WaitHandle)  
0e4eeeb0 65f5d68b System.Windows.Forms.Control.MarshaledInvoke(System.Windows.Forms.Control, System.Delegate, System.Object[], Boolean)  
0e4eef50 65c233ac System.Windows.Forms.Control.Invoke(System.Delegate, System.Object[])  
0e4eef84 65c2334f System.Windows.Forms.Control.Invoke(System.Delegate)  
0e4eef88 6715ad76 MyNameSpace.MyClass.MyMethod()  
0e4eefb0 67b96e96 System.Threading.ThreadHelper.ThreadStart_Context(System.Object)  
0e4eefbc 67ba031f System.Threading.ExecutionContext.Run(System.Threading.ExecutionContext, System.Threading.ContextCallback, System.Object)  
0e4eefd4 67b96e14 System.Threading.ThreadHelper.ThreadStart()  
0e4ef1fc 68f81b4c [GCFrame: 0e4ef1fc]   
4

3 回答 3

1

Win32(以及 .NET)中的许多操作都会阻塞。因为在所有阻塞 API 的底层实际上是一个带有等待的异步调用,所以当用户线程正在等待来自内核线程的某些东西时,这种等待调用是堆栈上的一个常见特性。

更新(现在我看到了你的调用堆栈):等待是因为调用的线程Invoke需要等待 UI 线程完成操作才能返回。WaitForMultipleObjectsEx是内核句柄上任何用户模式等待的底层等待 API(在这种情况下可能是一个事件对象)。

于 2011-11-25T17:13:17.420 回答
1

查看这篇文章这篇文章,它们会有所帮助,它们会向您展示不同 .NET 线程的所有正常调用,因此只需比较并找出问题所在..

编辑:
这篇精彩的文章展示了调用时调用 WaitOne API 的原因详细信息以及为什么会发生挂起。了解细节可能会帮助您避免这些挂起。

于 2011-11-25T17:19:21.460 回答
0

有很多 .Net 方法在内部执行等待。

例如,Control.Invoke将等待事件句柄,直到 UI 线程运行回调。

于 2011-11-25T17:12:15.650 回答