6

C里面有atexit函数,它

atexit() 函数注册要在正常进程终止时调用的给定函数,通过 exit(3) 或通过从程序的 main() 返回。

Python 也有类似的能力。

.NET 是否提供了在正常进程终止时调用代码的方法?我知道有一些类似DomainUnloadand的东西ProcessExit,但至少据我所知,这些是不可靠的——要么要求应用程序是 Windows 窗体(或 WPF 应用程序),要么是其他东西。我正在为 .dll 编写代码,所以我不能依赖诸如弄乱主程序功能之类的东西 - 将其包装在 try/catch 中。

我的最终目标是执行一些文件清理(即刷新缓冲区并关闭)。如果我可以调用一些非托管代码,例如 win32api 挂钩或其他东西,我完全可以接受。

4

1 回答 1

8

据我所知,没有直接的答案。

如果你想写一个健壮的 DLL,你应该准备几个场景:

  1. 您的代码托管在默认 AppDomain 中的 .NET 应用程序中。(琐碎的场景)
  2. 您的代码托管在由宿主代码创建的 AppDomain 中的 .NET 应用程序中。
  3. 您的代码托管在非托管应用程序(托管 CLR)中。

第三种情况是最难处理的,因为 CLR 可以被其主机禁用,因此托管代码将不再执行。

System.Windows.Forms.Application.ApplicationExit不好,因为它只适用于 WinForm 应用程序。

System.AppDomain.DomainUnload本身并不好,因为它永远不会为默认的 AppDomain 引发。

AppDomain.ProcessExit本身并不好:如果您的代码托管在单独的 AppDomain 中,则主机可能会卸载该 AppDomain,因此该事件永远不会引发。

我会首先尝试涵盖大多数情况,使用以下内容:

if (AppDomain.CurrentDomain.IsDefaultAppDomain())
    AppDomain.CurrentDomain.ProcessExit += MyTerminationHandler;
else
    AppDomain.CurrentDomain.DomainUnload += MyTerminationHandler;

但请注意以下备注(来自 MSDN):

所有 ProcessExit 事件处理程序的总执行时间都是有限的,就像所有终结器的总执行时间在进程关闭时受到限制一样。默认值为两秒。非托管主机可以通过使用 OPR_ProcessExit 枚举值调用 ICLRPolicyManager::SetTimeout 方法来更改此执行时间。

上面的代码仍然无人看管第三种情况。我知道有两种方法可以处理这种情况(以及前两种)

首先,可以使用System.Runtime.CompilerServices.RuntimeHelpers.ExecuteCodeWithGuaranteedCleanup方法,如下:

{
//  this goes at your code's entry point
    RuntimeHelpers.ExecuteCodeWithGuaranteedCleanup(MyExecutionCode, MyCleanupCode, null);
}
static void MyExecutionCode(object data) { /* your execution code here */}
static void MyCleanupCode(object data, bool exceptionThrown) { /* your cleanup code here */ }

其次,您可以通过继承该类并将清理代码放入终结器中来利用System.Runtime.ConstrainedExecution.CriticalFinalizerObject该类(请参阅此处的 MSDN )。这要求您的清理代码遵守受约束的执行区域指南。

于 2013-05-21T20:19:42.600 回答