Мы наконец-то узнали больше об этом (но к тому времени моя машина была восстановлена и потеряла файлы cookie в моем незарегистрированном профиле здесь; надеюсь, она позволит собранию опубликовать этот ответ).
Дальнейшее расследование в конечном итоге выявило еще несколько событий, которые мы сочли полезными:
System.Windows.Forms.Application.ThreadExit
- срабатывает при выходе из цикла сообщений
System.Windows.Forms.Application.ApplicationExit
- срабатывает, когда завершаются все циклы сообщений
System.AppDomain.CurrentDomain.DomainUnload
- Срабатывает, когда выходит домен, отличный от домена по умолчанию.
System.AppDomain.CurrentDomain.ProcessExit
- срабатывает при выходе из домена приложения по умолчанию
System.AppDomain.CurrentDomain.UnhandledException
- срабатывает при возникновении необработанного исключения, завершая работу приложения.
Только одно из событий DomainUnload
или ProcessExit
возможно для данного домена приложения, в зависимости от того, является ли он доменом по умолчанию (верхнего уровня) для процесса или был создан как поддомен (например, на веб сервер). Если приложение не знает, каким оно может быть (как в нашем случае), оно должно подписаться на оба, если оно хочет поймать фактическую выгрузку для себя. Кроме того, кажется, что UnhandledException
(который по состоянию на .NET2.0 всегда является фатальным) может предотвратить два других события, так что это может быть третий случай для обработки. Эти три события должны работать для любого приложения .NET.
Существует предупреждение о том, что время выполнения для ProcessExit
ограничено (около 4 секунд?), Поэтому, возможно, не удастся выполнить обширную "финальную" работу в этом обработчике событий. Это должно быть что-то, что можно сделать быстро.
События Application
применяются только к приложениям WinForms (однако мы подозреваем, что они могут не применяться в приложениях с чистым WPF). Названия могут вводить в заблуждение, потому что они названы для их основного обычного использования, которое имеет определенные предположения. ThreadExit
относится не к фактическому System.Threading.Thread
, а скорее к циклу сообщений (Application.Run()
)) потока пользовательского интерфейса, а ApplicationExit
аналогично относится к набору форм приложений в одном или нескольких потоках пользовательского интерфейса. Обычно, когда возвращается вызов Application.Run()
, вызванный методом ввода потока, метод ввода быстро завершается, и сам поток затем заканчивается. И как только все потоки пользовательского интерфейса закрываются, приложение WinForms обычно все готово и завершает работу.
Другим заметным событием является событие System.Windows.Forms.Application.ThreadException
. Цикл сообщений Windows может быть настроен на перехват исключений, возникающих при обработке сообщения, и отправку этого события, а не на то, чтобы они были неперехваченными (и, следовательно, фатальными) исключениями. Перехват этих исключений позволяет циклу сообщений (и тому потоку пользовательского интерфейса) продолжать работу (после прерывания текущего обработчика сообщений). В каждый момент времени для данного потока может быть только один подписчик на это событие (подписки перезаписывают любого предыдущего подписчика), и его необходимо настроить перед созданием любой формы и подпиской перед входом в цикл сообщений. См. Справку MSDN для этого события и System.Windows.Forms.Applicaton.SetUnhandledExceptionMode()
для получения дополнительной информации.