10

终结器总是由 .net 框架调用,因此序列可能会失控;即使构造函数失败,析构函数仍然可以被触发。

当此类终结器异常来自第三方库时,这可能会带来麻烦:我找不到处理它们的方法!

例如,在下面的代码中,虽然类 A 的构造函数总是抛出异常并失败,但 A 的终结器将由 .net 框架触发,并且由于 A 具有 B 类型的属性,所以调用 ~B()。

class Program // my code
{
    static void Main(string[] args)
    {
        A objA;
        try
        {
            objA = new A();
        }
        catch (Exception)
        {
        }

        ; // when A() throws an exception, objA is null

        GC.Collect(); // however, this can force ~A() and ~B() to be called.

        Console.ReadLine();
    }
}

public class A  // 3rd-party code
{
    public B objB;

    public A()
    {
        objB = new B(); // this will lead ~B() to be called.
        throw new Exception("Exception in A()");
    }

    ~A() // called by .net framework
    {
        throw new Exception("Exception in ~A()"); // bad coding but I can't modify
    } 
}

public class B // 3rd-party code
{
    public B() { }

    ~B() // called by .net framework
    {
        throw new Exception("Exception in ~B()"); // bad coding but I can't modify
    } 
}

如果这些是我的代码,那就更容易了——我可以在终结器中使用 try-catch,至少我可以做一些日志记录——我可以允许异常使程序崩溃,尽快发现错误——或者如果我想“容忍”异常,我可以有一个 try-catch 来抑制异常,并有一个优雅的退出。

但是如果 A 和 B 是来自 3rd-party 库的类,我什么也做不了!我无法控制异常的发生,我无法捕获它们,所以我无法记录或抑制它!

我能做些什么?

4

3 回答 3

2

听起来 3rd 方实用程序写得不好。:)

您是否尝试过使用AppDomain.UnhandledException捕获它?

于 2011-07-18T02:06:49.080 回答
0

您可能需要为您的应用程序考虑一个全局异常处理程序。您没有说明您是否在使用 ASP.NET、WinForm、MVC 等,但这里有一个控制台应用程序:

.NET 控制台应用程序中的全局异常处理程序

在 ASP.NET 中,您可以使用 Global.asax 文件来捕获未处理的异常。

如果您总是在应用程序中调用 GC.Collect(),您也可以尝试将其包装在 try-catch 块中。

只是一些需要考虑的想法。

于 2011-07-18T02:07:42.463 回答
0

您可以使用GC.SuppressFinalizer(objA)GC.KeepAlive(objA)这将阻止垃圾收集器调用该对象的 finalize,因此在使用KeepAliveobjB不会终结,因为objA“which is alive”仍然具有它的引用. 但是,如果您忘记以正确的方式完成或处置,您应该注意内存泄漏。objA

但是假设objA在某个方法中的某个点例如已经初始化了另一个objectB并且它没有正确地处理它,那么不幸的是我不认为你可以做任何事情。

您可以尝试的另一件事是检查当您处于Release模式而不是Debug模式时该库的行为是否有所不同;例如,如果在调试模式下调用终结器,它们可能只会在终结器处抛出异常“它对开发人员来说是一种帮助,所以如果他们没有以正确的方式调用 dispose 或终结对象,调试时将抛出异常”:

~A()
{
#if DEBUG
    throw new Exception("Exception in ~A()");
#endif//DEBUG
} 

如果不是这种情况,那么我认为您在处理该库时会遇到麻烦。

于 2011-07-18T03:03:58.953 回答