Search code examples
c#.nettask-parallel-libraryasync-awaitdelay

Task.Delay never completing


The following code will freeze forever.

public async Task DoSomethingAsync()
{
    await Task.Delay(2000);
}

private void Button_Click(object sender, RoutedEventArgs e)
{
    DoSomethingAsync().Wait();
    // Task.Delay(2000).Wait();
}

If I switch the call to DoSomethingAsync with the commented out code, it behaves as expected. I suspect that somehow the nested awaits are causing a deadlock, but I'm not sure why, or how to fix it.


Solution

  • Assuming Button_Click runs in the GUI thread you have a deadlock on your hands.

    When you use Wait on a task you are synchronously blocking the thread until the task ends, but the task will never end because the continuation (the completion of Task.Delay(2000);) must run on the GUI thread as well (which is blocked on Wait).

    You have several solutions. Either use ConfigureAwait(false) to not capture the GUI thread's SynchronizationContext:

    public async Task DoSomethingAsync()
    {
        await Task.Delay(2000).ConfigureAwait(false);
    }
    

    Or (which I recommend) use an async void event handler (which is the only appropriate place for an async void method):

    private async void Button_Click(object sender, RoutedEventArgs e)
    {
        await DoSomethingAsync();
    }
    
    public async Task DoSomethingAsync()
    {
        await Task.Delay(2000);
    }