我有一个窗口表单应用程序,它有多个运行的线程,可以在主UI线程上调用以更新UI。有时在开发机器上,应用程序主UI线程将停止运行,应用程序不再响应。如果我让应用程序一夜之间运行似乎就会发生。但是,我有通过远程桌面运行此窗口表单应用程序的用户,如果应用程序在没有用户交互的情况下一夜之间运行,则会出现此问题。
我发现article似乎在描述这个问题,但我没有足够的Windows开发知识来弄清楚应用程序冻结的原因。
我得到的唯一信息是以下堆栈跟踪,表明主UI线程正在等待某种操作。
这个问题一直困扰着我很长一段时间。我将不胜感激任何建议或意见。
谢谢!
Main UI thread stack trace: mscorlib.dll!System.Threading.WaitHandle.WaitOne(long timeout, bool exitContext) + 0x2f bytes mscorlib.dll!System.Threading.WaitHandle.WaitOne(int millisecondsTimeout, bool exitContext) + 0x25 bytes System.Windows.Forms.dll!System.Windows.Forms.Control.WaitForWaitHandle(System.Threading.WaitHandle waitHandle = {System.Threading.ManualResetEvent}) Line 4268 C# System.Windows.Forms.dll!System.Windows.Forms.Control.MarshaledInvoke(System.Windows.Forms.Control caller, System.Delegate method, object[] args, bool synchronous) Line 7614 C# System.Windows.Forms.dll!System.Windows.Forms.Control.Invoke(System.Delegate method, object[] args) Line 7178 + 0x11 bytes C# System.Windows.Forms.dll!System.Windows.Forms.WindowsFormsSynchronizationContext.Send(System.Threading.SendOrPostCallback d, object state) Line 89 C# System.dll!Microsoft.Win32.SystemEvents.SystemEventInvokeInfo.Invoke(bool checkFinalization = true, object[] args = {object[2]}) + 0x62 bytes System.dll!Microsoft.Win32.SystemEvents.RaiseEvent(bool checkFinalization = true, object key = {object}, object[] args = {object[2]}) + 0x10f bytes System.dll!Microsoft.Win32.SystemEvents.OnUserPreferenceChanging(int msg, System.IntPtr wParam, System.IntPtr lParam) + 0x77 bytes System.dll!Microsoft.Win32.SystemEvents.WindowProc(System.IntPtr hWnd = 2032836, int msg = 8218, System.IntPtr wParam = 47, System.IntPtr lParam = 100019840) + 0x2ca bytes [Native to Managed Transition] [Managed to Native Transition] System.Windows.Forms.dll!System.Windows.Forms.Application.ComponentManager.System.Windows.Forms.UnsafeNativeMethods.IMsoComponentManager.FPushMessageLoop(int dwComponentID, int reason = 4, int pvLoopData = 0) Line 2106 + 0x8 bytes C# System.Windows.Forms.dll!System.Windows.Forms.Application.ThreadContext.RunMessageLoopInner(int reason = 4, System.Windows.Forms.ApplicationContext context = {System.Windows.Forms.Application.ModalApplicationContext}) Line 3377 + 0x1b bytes C# System.Windows.Forms.dll!System.Windows.Forms.Application.ThreadContext.RunMessageLoop(int reason, System.Windows.Forms.ApplicationContext context) Line 3261 + 0xa bytes C# System.Windows.Forms.dll!System.Windows.Forms.Application.RunDialog(System.Windows.Forms.Form form) Line 1488 C# System.Windows.Forms.dll!System.Windows.Forms.Form.ShowDialog(System.Windows.Forms.IWin32Window owner) Line 6120 + 0x8 bytes C# Schedule.exe!ME.APTS.ScheduleApp.ScheduleAppMainForm.ShowOpenScheduleForm.AnonymousMethod() Line 829 + 0xd bytes C# Schedule.exe!ME.APTS.ScheduleApp.ScheduleAppMainForm.PromptUserToSaveSchedule(System.Action oAfterPromptUserToSaveCallBack = {Method = Cannot evaluate expression because the code of the current method is optimized.}) Line 1858 + 0xb bytes C# Schedule.exe!ME.APTS.ScheduleApp.ScheduleAppMainForm.ShowOpenScheduleForm() Line 859 + 0xb bytes C# [Native to Managed Transition] [Managed to Native Transition] mscorlib.dll!System.Delegate.DynamicInvokeImpl(object[] args) + 0x55 bytes System.Windows.Forms.dll!System.Windows.Forms.Control.InvokeMarshaledCallbackDo(System.Windows.Forms.Control.ThreadMethodEntry tme) Line 7266 + 0xb bytes C# System.Windows.Forms.dll!System.Windows.Forms.Control.InvokeMarshaledCallbackHelper(object obj) Line 7228 + 0x7 bytes C# mscorlib.dll!System.Threading.ExecutionContext.runTryCode(object userData) + 0x51 bytes [Native to Managed Transition] [Managed to Native Transition] mscorlib.dll!System.Threading.ExecutionContext.RunInternal(System.Threading.ExecutionContext executionContext, System.Threading.ContextCallback callback, object state) + 0x67 bytes mscorlib.dll!System.Threading.ExecutionContext.Run(System.Threading.ExecutionContext executionContext, System.Threading.ContextCallback callback, object state) + 0x45 bytes System.Windows.Forms.dll!System.Windows.Forms.Control.InvokeMarshaledCallback(System.Windows.Forms.Control.ThreadMethodEntry tme) Line 7213 + 0xffffffc5 bytes C# System.Windows.Forms.dll!System.Windows.Forms.Control.InvokeMarshaledCallbacks() Line 7297 + 0xb bytes C# System.Windows.Forms.dll!System.Windows.Forms.Control.WndProc(ref System.Windows.Forms.Message m) Line 13848 C# System.Windows.Forms.dll!System.Windows.Forms.ScrollableControl.WndProc(ref System.Windows.Forms.Message m) Line 1491 C# System.Windows.Forms.dll!System.Windows.Forms.ContainerControl.WndProc(ref System.Windows.Forms.Message m) Line 1898 C# System.Windows.Forms.dll!System.Windows.Forms.Form.WndProc(ref System.Windows.Forms.Message m) Line 7515 C# System.Windows.Forms.dll!System.Windows.Forms.Control.ControlNativeWindow.OnMessage(ref System.Windows.Forms.Message m) Line 14051 C# System.Windows.Forms.dll!System.Windows.Forms.Control.ControlNativeWindow.WndProc(ref System.Windows.Forms.Message m) Line 14106 C# System.Windows.Forms.dll!System.Windows.Forms.NativeWindow.Callback(System.IntPtr hWnd, int msg = 49512, System.IntPtr wparam, System.IntPtr lparam) Line 647 + 0xa bytes C# System.Windows.Forms.dll!System.Windows.Forms.NativeWindow.DefWndProc(ref System.Windows.Forms.Message m = {System.Windows.Forms.Message}) Line 814 + 0x1d bytes C# System.Windows.Forms.dll!System.Windows.Forms.NativeWindow.WndProc(ref System.Windows.Forms.Message m) Line 1409 C# Infragistics2.Win.UltraWinToolbars.v8.1.dll!Infragistics.Win.UltraWinToolbars.UltraToolbarsManager.FormSubClasser.WndProcImpl(ref System.Windows.Forms.Message m) + 0x17f5 bytes Infragistics2.Win.UltraWinToolbars.v8.1.dll!Infragistics.Win.UltraWinToolbars.UltraToolbarsManager.FormSubClasser.WndProc(ref System.Windows.Forms.Message m) + 0x5 bytes System.Windows.Forms.dll!System.Windows.Forms.NativeWindow.Callback(System.IntPtr hWnd, int msg = 49512, System.IntPtr wparam, System.IntPtr lparam) Line 647 + 0xa bytes C# [Native to Managed Transition] [Managed to Native Transition] System.Windows.Forms.dll!System.Windows.Forms.Application.ComponentManager.System.Windows.Forms.UnsafeNativeMethods.IMsoComponentManager.FPushMessageLoop(int dwComponentID, int reason = -1, int pvLoopData = 0) Line 2106 + 0x8 bytes C# System.Windows.Forms.dll!System.Windows.Forms.Application.ThreadContext.RunMessageLoopInner(int reason = -1, System.Windows.Forms.ApplicationContext context = {System.Windows.Forms.ApplicationContext}) Line 3377 + 0x1b bytes C# System.Windows.Forms.dll!System.Windows.Forms.Application.ThreadContext.RunMessageLoop(int reason, System.Windows.Forms.ApplicationContext context) Line 3261 + 0xa bytes C# System.Windows.Forms.dll!System.Windows.Forms.Application.Run() Line 1457 C# Schedule.exe!ME.APTS.ScheduleApp.ScheduleApp.LoadData() Line 318 + 0x5 bytes C# Schedule.exe!ME.APTS.ScheduleApp.ScheduleApp.Run() Line 170 + 0x9 bytes C# Schedule.exe!ME.APTS.ScheduleApp.ScheduleApp.Main() Line 126 + 0xb bytes C#
答案 0 :(得分:13)
是的,这是一个由SystemEvents类引起的相当臭名昭着的线程问题。我从来没有得到过可靠的诊断,但90%的可能性是由你的应用程序中的初始化问题引发的。
根本问题是,SystemEvents会根据您应用中的第一个表单按需初始化,该表单具有对其生成的事件感兴趣的控件。如果第一个表单是而不是在主线程中创建,那么SystemEvents无法猜测哪个线程是程序中的UI线程。最终,当收到通知(如UserPreferenceChanging)时,它会尝试在该线程上触发事件,但它不再存在。 SynchronizationContext类中的回退代码会在线程池线程上引发事件。这不可避免地通过在没有创建窗口的线程上运行UI代码来调用Threading Hell。当发生这种情况时,很多事情都会出错。在工作站锁定后恢复桌面时,死锁是一种特别常见的结果。
这不是唯一可能出错的方法,如果你在另一个线程上创建任何表单,这是不可避免的。现在SystemEvents当然不可能在正确的线程上引发事件,有人会输。一篇博客文章,演示了调试技术is here。是的,丑陋的。理想情况下,控件知道处理此问题并编组通知本身。但是,在.NET 2.0,DataGridView,NumericUpDown,DomainUpDown,ToolStrip + MenuStrip和ToolStripItem派生类中,这是遗忘的知识。我应该注意到RichTextBox和ProgressBar是可疑的,其余的都没问题。
查看应用的启动顺序。创建自己的初始屏幕是一个很好的领导,请使用WindowsFormsApplicationBase类提供的内置支持。如果你自己做,那么保持它非常简单,只是一个位图。如上所述,您可以在工作线程上创建自己的表单的任何地方都会遇到麻烦。总是这样做,在工作者上运行昂贵的代码并将UI保留在主线程上。
答案 1 :(得分:7)
大约一年前我遇到了同样的问题(应用程序在没有用户交互的情况下挂起一段时间后,调用堆栈中有OnUserPreferenceChanging()
)。
最可能的原因是您在控件上使用InvokeRequired
/ Invoke()
而不在主窗体上。如果尚未创建控件的句柄,这有时会产生错误的结果。
解决方案是始终在主窗口上调用InvokeRequired
/ Invoke()
(如果您不想向表单类引入依赖项,可以将其转换为ISynchronizeInvoke
)。
您可以找到关于原因和解决方案here的优秀,非常详细的描述。
答案 2 :(得分:2)
我遇到了同样的问题,这总是由于Microsoft.Win32.SystemEvents.DisplaySettingsChanged事件在Windows 8.1下以及我的应用程序运行时以及与VNC或RDP连接的人更频繁地发生。使用Windows x.x和Mac上的Fusion(VMWare)时,也会非常清楚地更改桌面设置。
在尝试了很多东西后,我终于通过在我的MainApp中监听这些事件来解决它(创建所有对话框并执行所有调用的那个)
宣告:
Microsoft.Win32.SystemEvents.DisplaySettingsChanged += SystemEvents_DisplaySettingsChanged;
Microsoft.Win32.SystemEvents.DisplaySettingsChanging += SystemEvents_DisplaySettingsChanging;
Microsoft.Win32.SystemEvents.UserPreferenceChanged += SystemEvents_UserPreferenceChanged;
实施:
static void SystemEvents_UserPreferenceChanged(object sender, Microsoft.Win32.UserPreferenceChangedEventArgs e)
{
//Do nothing
}
static void SystemEvents_DisplaySettingsChanged(object sender, EventArgs e)
{
//Do nothing
}
static void SystemEvents_DisplaySettingsChanging(object sender, EventArgs e)
{
//Do nothing
}
捕获这些事件什么都不做,但这似乎排除了当我从Windows发出这些事件并且我的代码的任何其他部分等待MainApp参加Invoke时我遇到的僵局。
希望这有帮助。
答案 3 :(得分:0)
请检查this答案中的代码,该代码可以帮助您找到导致Windows Form应用程序因SystemEvents类的线程问题而冻结的确切事件处理程序。