Nie do końca rozumiem różnicę między Task.Waiti await.
Mam coś podobnego do następujących funkcji w usłudze WebAPI ASP.NET:
public class TestController : ApiController
{
public static async Task<string> Foo()
{
await Task.Delay(1).ConfigureAwait(false);
return "";
}
public async static Task<string> Bar()
{
return await Foo();
}
public async static Task<string> Ros()
{
return await Bar();
}
// GET api/test
public IEnumerable<string> Get()
{
Task.WaitAll(Enumerable.Range(0, 10).Select(x => Ros()).ToArray());
return new string[] { "value1", "value2" }; // This will never execute
}
}
Gdzie Getbędzie impas.
Co może to powodować? Dlaczego nie powoduje to problemów, gdy używam oczekiwania blokującego zamiast await Task.Delay?
Task.Delay(1).Wait()jest w zasadzie dokładnie tym samym co Thread.Sleep(1000). W rzeczywistym kodzie produkcyjnym rzadko jest to właściwe.
WaitAllpowoduje impas. Zobacz link do mojego bloga w mojej odpowiedzi, aby uzyskać więcej informacji. Zamiast tego powinieneś użyć await Task.WhenAll.
ConfigureAwait(false)do pojedynczego połączenia do Barlub Rosnie impasu, ale ponieważ masz przeliczalny, który jest tworzenie więcej niż jeden, a następnie czeka na wszystkich tych, pierwszy pasek impasu drugi. Jeśli await Task.WhenAllzamiast czekać na wszystkie zadania, aby nie zablokować kontekstu ASP, zobaczysz, że metoda powraca normalnie.
.ConfigureAwait(false) całego drzewa w górę, aż do momentu zablokowania, w ten sposób nic nigdy nie próbuje wrócić do głównego kontekstu; to powinno działać. Inną opcją byłoby rozwinięcie wewnętrznego kontekstu synchronizacji. Link . Jeśli umieścisz Task.WhenAllw sposób AsyncPump.Runzostanie ona skutecznie blokować na całość bez potrzeby ConfigureAwaitgdziekolwiek, ale to prawdopodobnie nadmiernie skomplikowane rozwiązanie.
Task.Delay(1).Wait()co jest wystarczająco dobry.