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 に答える