C# WinForms 托盘应用程序 - 未捕获 windows 关闭事件

C# WinForms Tray application - not catching windows shutdown events

我正在编写一个监视状态的程序。它启动 main window (LoginForm) 以询问用户凭据,然后隐藏表单。在 LoginForm 初始化 NotifyIcon 之后,所有剩余的工作都在隐藏 LoginForm 的情况下完成。我已经在 LoginForm 的 FormClosing 事件中实现了所有清理工作。在正常退出过程中,一切正常。

问题是程序一直位于托盘中,我往往会忘记在关闭前退出它windows - 程序不会在关闭事件时保存状态。

我已经阅读了许多论坛和文档,并且根据我所阅读的内容,事件 FormClosing/FormClosed + SessionEnding/SessionEnded 无论如何都必须触发。但看起来他们就是不开火。甚至无法在 SessionEnding 中取消关闭(愚蠢的 e.Cancel=true)- 程序没有到达那个地方就消失了。

我为自己制作了一个小型调试库,用于将调试信息写到一个文件中,并立即刷新缓存。我已将调试消息添加到所有事件。当我尝试在 Windows 中注销用户(与关闭相同)进行测试时 - 我通常看不到任何事件被触发,只是程序消失了。没有我可以捕捉到的异常,没有事件等。当手动或通过 "taskkill /IM " 退出程序时 - 我看到所有预期的调试打印输出。更有趣的是 - 有时程序确实会到达关闭事件并在 Windows 关闭期间进行适当的清理工作。

已经在这上面浪费了几天时间。按照 MS 文章中的建议覆盖 WndProc - 程序也不会到达 WndProc(以确保它在关闭表单之前先出现)。尝试代替 FormClosing 事件来覆盖 LoginForm 的 OnClosing - 运气不好。添加了 UnhandledException 处理程序、Microsoft.Win32.SessionEnding、Microsoft.Win32.SessionEnded、Form.FormClosing、Form.FormClosed 事件处理程序 - 运气不好。

我怀疑这要么是隐藏表单的问题,要么是仅在关闭期间发生了某种异常(资源已处理?)。我怎样才能找到为什么会这样?是否有一些简单的方法可以模拟单个应用程序的 Windows 关闭,以便在 VS 中进行调试?我已经尝试过 RMTool - 由于某种原因它无法模拟关闭并且程序只是忽略它。

更新:程序使用System.Timers.Timer定期轮询服务器是否有任何更改。

我认为 windows 只是配置为自动关闭所有应用程序,因此应用程序没有机会捕获 SessionEnding 事件等:看看 http://www.addictivetips.com/windows-tips/disable-automatic-termination-of-applications-during-shutdown-in-windows-7/ 或检查您的配置计算机配置 > 管理模板 > 系统 > 关机选项。

我对此做了一些研究,基本上,在 Windows XP 之后,他们改变了处理关机的方式。

您无法使用 form_closing 事件等可靠地阻止或捕获关闭事件

您必须使用新的 API 才能这样做。这里有一个完整的例子:http://bartdesmet.net/blogs/bart/archive/2006/10/25/Windows-Vista-2D00-ShutdownBlockReasonCreate-in-C_2300_.aspx

您应该可以调用该函数来阻止关闭(显示 'saving changes...' 等消息),然后在后台保存并退出您的应用程序。一旦您的程序退出,它应该允许 windows 继续关闭。