Task does not wait for ContinueWith to finish

前端 未结 3 498
南旧
南旧 2021-01-19 16:42

I have console application and code as below,

My problem is before ContinueWith task finish, the console application ends, it does not waits the continueWith to fini

相关标签:
3条回答
  • 2021-01-19 17:16

    I tried another solution that worked for me :

    try {
        taskNotOnFaulted.Wait();
    } catch {
        taskOnlyOnFaulted.Wait();
    }
    
    0 讨论(0)
  • 2021-01-19 17:26

    Jorge's solution is not working when an exception is thrown :

    var task = new Task(() =>
        {
            Console.WriteLine("My task...");
            throw new Exception();
        });
    
    task.Start();
    
    var taskNotOnFaulted = task.ContinueWith(t =>
    {
        Thread.Sleep(1000);
        Console.WriteLine("NotOnFaulted");
    }, TaskContinuationOptions.NotOnFaulted | TaskContinuationOptions.ExecuteSynchronously);
    var taskOnlyOnFaulted = task.ContinueWith(t =>
    {
        Thread.Sleep(1000);
        Console.Write("OnlyOnFaulted");
    }, TaskContinuationOptions.OnlyOnFaulted | TaskContinuationOptions.ExecuteSynchronously);
    
    Task.WaitAny(taskNotOnFaulted, taskOnlyOnFaulted);
    
    Console.WriteLine("Finished");
    

    The output is :

    My task...
    Finished
    OnlyOnFaulted
    

    This is because the taskNotOnFaulted get a Cancelled status when the exception is thrown, whereas it keeps a WaitingForActivation status when no exception is thrown.

    So you have to replace :

    Task.WaitAny(taskNotOnFaulted, taskOnlyOnFaulted);
    

    by

    if (task.IsFaulted)
        taskOnlyOnFaulted.Wait();
    else
        taskNotOnFaulted.Wait();
    
    0 讨论(0)
  • 2021-01-19 17:33

    You have to wait on the task to complete from the main thread. Simplified it'll look like

    var task1 = Task<bool>.Factory.StartNew(() => DoProcess());
    
    successContinuation = task1 .ContinueWith(t1 => updateSuccess(),
                                              TaskContinuationOptions.NotOnFaulted | TaskContinuationOptions.ExecuteSynchronously)
    failureContinuation = task1 .ContinueWith( t => updateFault(),
                                              TaskContinuationOptions.OnlyOnFaulted | TaskContinuationOptions.ExecuteSynchronously);
    
    Task.WaitAny(successContinuation, failureContinuation);
    

    I think you're not realizing that you're creating a task that will (actually just may) execute in a different thread. The moment you start your task you have two different threads of execution the main thread, which will continue to run, and your task. If you want your main thread (your console application) to wait for the task to finish, you have to manually specify it.

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