简体   繁体   中英

Is it safe to use async/await in ASP.NET event handlers?

I was doing some coding in ASP.NET when I came across this:

protected async void someButtonClickHandler(...)
{
    if(await blah)
        doSomething();
    else
        doSomethingElse();
}

After asking this question I got a better understanding of how async / await works. But then it struck me is it safe to use async / await in the manner shown above?
I mean after calling await blah the caller continues execution. Which means it might render the response back to client before await blah completes. Is this right? And if this is the case what happens to doSomething() / doSomethingElse() . Will they ever be executed? If they're executed does the user see the effects of their changes?
In my case these methods change some data displayed to the user, but I'd also like to know what would happen in the general case.

Yes it is safe, but not really recommended. The recommended way to do this is via RegisterAsyncTask . However, ASP.NET (Web Forms) will correctly handle async void event handlers.

The response is not rendered to the client when the handler await s; the await only yields to the ASP.NET runtime, not to the client. The ASP.NET runtime is aware that the event handler has not completed, so it knows not to send the response. When the event handler completes, the ASP.NET runtime responds by sending the response at that time.

I have an MSDN article on async ASP.NET that you may find helpful. If you're curious about how the ASP.NET runtime is aware that the async handler has not completed, I cover that in an earlier MSDN article .

I agree with Stephens answer (in short the ASP.NET SynchronisationContext keeps watch on how many tasks are running), but because it is a (best avoided) async void you might want to log any exceptions that would otherwise go un-noticed:

protected async void someButtonClickHandler(...)
{
    try{
        await someButtonClickHandlerInner(...)
    }
    catch (AggregateException ex)
    {
        logger.log(ex.flatten());
    }
    catch(Exception e){
        logger.log(e);  
    }
}

private async Task someButtonClickHandlerInner(...){
    if(await blah)
        doSomething();
    else
        doSomethingElse();
}

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM