Así que estoy tratando de entender esta nueva 'asincronización' en .NET 4.5. Me previamente jugado un poco con los controladores asincrónicos y la Biblioteca paralelo de tareas y terminó con este pedazo de código:MVC4 asincrónico y ejecución paralela
Tome este modelo:
public class TestOutput
{
public string One { get; set; }
public string Two { get; set; }
public string Three { get; set; }
public static string DoWork(string input)
{
Thread.Sleep(2000);
return input;
}
}
que se utiliza en un controlador de la siguiente manera:
public void IndexAsync()
{
AsyncManager.OutstandingOperations.Increment(3);
Task.Factory.StartNew(() =>
{
return TestOutput.DoWork("1");
})
.ContinueWith(t =>
{
AsyncManager.OutstandingOperations.Decrement();
AsyncManager.Parameters["one"] = t.Result;
});
Task.Factory.StartNew(() =>
{
return TestOutput.DoWork("2");
})
.ContinueWith(t =>
{
AsyncManager.OutstandingOperations.Decrement();
AsyncManager.Parameters["two"] = t.Result;
});
Task.Factory.StartNew(() =>
{
return TestOutput.DoWork("3");
})
.ContinueWith(t =>
{
AsyncManager.OutstandingOperations.Decrement();
AsyncManager.Parameters["three"] = t.Result;
});
}
public ActionResult IndexCompleted(string one, string two, string three)
{
return View(new TestOutput { One = one, Two = two, Three = three });
}
Este controlador muestra la vista en 2 segundos, gracias a la magia del TPL.
Ahora que esperaba (ingenuamente) que el código anterior se traduciría en la siguiente, utilizando el nuevo 'asíncrono' y 'esperar' características de C# 5:
public async Task<ActionResult> Index()
{
return View(new TestOutput
{
One = await Task.Run(() =>TestOutput.DoWork("one")),
Two = await Task.Run(() =>TestOutput.DoWork("two")),
Three = await Task.Run(() =>TestOutput.DoWork("three"))
});
}
Este controlador representa la vista en 6 segundos. En algún lugar de la traducción, el código dejó de ser paralelo. Sé que async y paralelo son dos conceptos diferentes, pero de alguna manera pensé que el código funcionaría de la misma manera. ¿Podría alguien señalar qué está sucediendo aquí y cómo se puede solucionar?
Así que básicamente mezclé la asincronía y el paralelismo. I (erróneamente) pensé que la espera real ocurriría cuando el resultado de una tarea se utilizara finalmente (no muy diferente de como una consulta linq solo se ejecuta cuando se enumera). – Lodewijk