Async call with await in HttpClient never returns

后端 未结 3 535
栀梦
栀梦 2020-11-29 19:33

I have a call I am making from inside a xaml-based, C# metro application on the Win8 CP; this call simply hits a web service and returns JSON data.



        
相关标签:
3条回答
  • 2020-11-29 20:01

    Check out this answer to my question which seems to be very similar.

    Something to try: call ConfigureAwait(false) on the Task returned by GetStreamAsync(). E.g.

    var result = await httpClient.GetStreamAsync("weeklyplan")
                                 .ConfigureAwait(continueOnCapturedContext:false);
    

    Whether or not this is useful depends on how your code above is being called - in my case calling the async method using Task.GetAwaiter().GetResult() caused the code to hang.

    This is because GetResult() blocks the current thread until the Task completes. When the task does complete it attempts to re-enter the thread context in which it was started but cannot because there is already a thread in that context, which is blocked by the call to GetResult()... deadlock!

    This MSDN post goes into a bit of detail on how .NET synchronizes parallel threads - and the answer given to my own question gives some best practices.

    0 讨论(0)
  • 2020-11-29 20:05

    Just a heads up - if you miss the await at the top level in an ASP.NET controller, and you return the task instead of the result as a response, it actually just hangs in the nested await call(s) with no errors. A silly mistake, but had I seen this post it might have saved me some time checking through the code for something odd.

    0 讨论(0)
  • 2020-11-29 20:26

    Disclaimer: I don't like the ConfigureAwait() solution because I find it non-intuitive and hard to remember. Instead I came to the conclusion to wrap non awaited method calls in Task.Run(() => myAsyncMethodNotUsingAwait()). This seems to work 100% but might just be a race condition!? I'm not so sure what is going on to be honest. This conclusion might be wrong and I risk my StackOverflow points here to hopefully learn from the comments :-P. Please do read them!

    I just had the problem as described and found more info here.

    The statement is: "you can't call an asynchronous method"

    await asyncmethod2()
    

    from a method that blocks

    myAsyncMethod().Result
    

    In my case I couldn't change the calling method and it wasn't async. But I didn't actually care about the result. As I remember it also didn't work removing the .Result and have the await missing.

    So I did this:

    public void Configure()
    {
        var data = "my data";
        Task.Run(() => NotifyApi(data));
    }
    
    private async Task NotifyApi(bool data)
    {
        var toSend = new StringContent(JsonConvert.SerializeObject(data), Encoding.UTF8, "application/json");
        await client.PostAsync("http://...", data);
    }
    

    In my case I didn't care about the result in the calling non-async method but I guess that is quite common in this use case. You can use the result in the calling async method.

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