Can ThreadAbortException skip finally?

后端 未结 3 908
花落未央
花落未央 2020-12-11 06:43

Everything I\'ve read claims an abort on a thread will execute the finally block before ending from a ThreadAbortException. I wanted to confirm this so I can plan on how to

相关标签:
3条回答
  • 2020-12-11 07:19

    The finally shouldn't normally be skipped.

    It's possible that the console app (assuming it is one) is exiting before the finally block runs (since there is no wait after you call Thread.Abort()).

    What happens if you put a Console.ReadLine() right at the end of the program?

    0 讨论(0)
  • 2020-12-11 07:27

    Docs say: ThreadAbortException is a special exception that can be caught, but it will automatically be raised again at the end of the catch block. When this exception is raised, the runtime executes all the finally blocks before ending the thread. Because the thread can do an unbounded computation in the finally blocks or call Thread.ResetAbort to cancel the abort, there is no guarantee that the thread will ever end.

    I'm pretty sure your thread is being dumped because you exit the method and lose a reference to it, and so it gets collected by the Garbage Collector. Try making the testThread variable a field member of the class and see what happens.

    That, or you have a race condition since the threads run in parallel: the main thread is finishing before the spun-up test thread can output the finally data (Exceptions are expensive and take time to reach catch or finally blocks).

    0 讨论(0)
  • 2020-12-11 07:39

    The finally block in the worker thread function is executed on the worker thread which is parallel to the main thread. It's a race condition. You can not tell which of worker thread finally or main thread code after abort call get executed sooner. If you need a synchronous abort then you have to put something like that:

            if (testThread.IsAlive)
            {
                testThread.Abort();
    
                bool blnFinishedAfterAbort = testThread.Join(TimeSpan.FromMilliseconds(1000));
                if (!blnFinishedAfterAbort)
                {
                    Console.WriteLine("Thread abort failed.");
                }
            }
            Console.WriteLine("main thread after abort call " + DateTime.Now.ToShortTimeString());
    

    Keep in mind that if you have legacy exception handling enabled (see http://msdn.microsoft.com/en-us/library/ms228965.aspx) and you specified the AppDomain_UnahandledException event handler then the ThreadAbortException will lead execution to that handler BEFORE the finally block in the worker thread function. This is just another example of frustrating and unexpected execution order one should be aware of while aborting threads.

    0 讨论(0)
提交回复
热议问题