2

我有一个 System.Timers.Timer,我发现在 ElapsedEventHandler 中有时会发生异常,但进程并没有死,但是执行 ElapsedEventHandler 的线程在发生异常的地方完全死了。不好,因为那时我不知道异常。为什么进程没有终止?如果 ElapsedEventHandler 中发生异常,如何以最佳方式终止进程?

我尝试使用 System.Threading.Timer 进行类似操作,然后进程终止。你能解释一下区别吗?

完整代码如下:

class Program
{
    static public void DoBad()
    {
        Console.WriteLine("DoBad enter");
        string s = "hello";
        object o = s;
        int i = (int)o; // Ops! Something unexpected happend here... Exception
        Console.WriteLine("DoBad exit");
    }

    static void DoIt1(object sender, System.Timers.ElapsedEventArgs e)
    {
        Console.WriteLine("\nDoIt1 enter");
        DoBad(); // Process is NOT terminated when excpetion occurs in DoBad, thread just dies quietly but process continues. Why is process not terminated?
        Console.WriteLine("DoIt1 exit");
    }

    static void Test1()
    {
        System.Timers.Timer t = new System.Timers.Timer(2000);
        t.Elapsed += new System.Timers.ElapsedEventHandler(DoIt1);
        t.Enabled = true;
    }

    static public void DoIt2(Object o)
    {
        Console.WriteLine("\nDoIt2 enter");
        DoBad(); // Process terminated when exception occurs in DoBad.
        Console.WriteLine("DoIt2 exit");
    }

    static void Test2()           
    {
        System.Threading.Timer timer = new System.Threading.Timer(DoIt2, null, 0, 2000);
    }

    static void Main(string[] args)
    {
        Test1(); // Process NOT terminated when exception occurs in DoBad
        // Test2(); // Process terminated when exception occurs in DoBad

        // Why is process NOT terminated when running Test1 (but terminated when running Test2)?
        // Want process to die. How to do it in best way?

        Console.WriteLine("Main Sleep");
        System.Threading.Thread.Sleep(11000);
        Console.WriteLine("Main Slept");
    }
}

/ 谢谢!

4

1 回答 1

0

这是按设计完成的,您可以在此处看到: ( system.timers.timer )
在 .NET Framework 2.0 版和更早版本中,Timer 组件捕获并抑制 Elapsed 事件的事件处理程序引发的所有异常。此行为可能会在 .NET Framework 的未来版本中发生更改。

system.threading.timer 不这样做,因此异常被传递并终止进程。如果您想了解异常,那么您应该捕获并记录它。
如果您希望所有工作都在主线程中完成,您可以使用 invoke/begininvoke:

form1.Invoke((ThreadStart)delegate
                {
                //do something
                });

或使用 Windows.forms.timer (winforms) / System.Windows.Threading.DispatcherTimer (WPF)

于 2012-11-26T15:32:46.533 回答