Me gustaría preguntarle su opinión sobre la arquitectura correcta cuando usarla Task.Run
. Estoy experimentando una interfaz de usuario lenta en nuestra aplicación WPF .NET 4.5 (con el marco Caliburn Micro).
Básicamente estoy haciendo (fragmentos de código muy simplificados):
public class PageViewModel : IHandle<SomeMessage>
{
...
public async void Handle(SomeMessage message)
{
ShowLoadingAnimation();
// Makes UI very laggy, but still not dead
await this.contentLoader.LoadContentAsync();
HideLoadingAnimation();
}
}
public class ContentLoader
{
public async Task LoadContentAsync()
{
await DoCpuBoundWorkAsync();
await DoIoBoundWorkAsync();
await DoCpuBoundWorkAsync();
// I am not really sure what all I can consider as CPU bound as slowing down the UI
await DoSomeOtherWorkAsync();
}
}
De los artículos / videos que leí / vi, sé que await
async
no necesariamente se ejecuta en un hilo de fondo y para comenzar a trabajar en segundo plano, debe envolverlo con esperar Task.Run(async () => ... )
. El uso async
await
no bloquea la interfaz de usuario, pero aún se está ejecutando en el subproceso de la interfaz de usuario, por lo que la hace lenta.
¿Dónde está el mejor lugar para poner Task.Run?
Debería solo
Ajuste la llamada externa porque esto es menos trabajo de subprocesamiento para .NET
, ¿o debería ajustar solo los métodos vinculados a la CPU que se ejecutan internamente,
Task.Run
ya que esto lo hace reutilizable para otros lugares? No estoy seguro aquí si comenzar a trabajar en subprocesos de fondo en el núcleo es una buena idea.
Anuncio (1), la primera solución sería así:
public async void Handle(SomeMessage message)
{
ShowLoadingAnimation();
await Task.Run(async () => await this.contentLoader.LoadContentAsync());
HideLoadingAnimation();
}
// Other methods do not use Task.Run as everything regardless
// if I/O or CPU bound would now run in the background.
Anuncio (2), la segunda solución sería así:
public async Task DoCpuBoundWorkAsync()
{
await Task.Run(() => {
// Do lot of work here
});
}
public async Task DoSomeOtherWorkAsync(
{
// I am not sure how to handle this methods -
// probably need to test one by one, if it is slowing down UI
}
fuente
await Task.Run(async () => await this.contentLoader.LoadContentAsync());
simplemente debería serawait Task.Run( () => this.contentLoader.LoadContentAsync() );
. AFAIK no ganas nada agregando un segundoawait
yasync
adentroTask.Run
. Y como no estás pasando parámetros, eso se simplifica un poco másawait Task.Run( this.contentLoader.LoadContentAsync );
.Respuestas:
Tenga en cuenta las pautas para realizar el trabajo en un hilo de la interfaz de usuario , recopiladas en mi blog:
Hay dos técnicas que debes usar:
1) Úselo
ConfigureAwait(false)
cuando pueda.Por ejemplo, en
await MyAsync().ConfigureAwait(false);
lugar deawait MyAsync();
.ConfigureAwait(false)
le indicaawait
que no necesita reanudar en el contexto actual (en este caso, "en el contexto actual" significa "en el hilo de la interfaz de usuario"). Sin embargo, para el resto de eseasync
método (después delConfigureAwait
), no puede hacer nada que suponga que está en el contexto actual (por ejemplo, actualizar elementos de la interfaz de usuario).Para obtener más información, consulte mi artículo de MSDN Mejores prácticas en programación asincrónica .
2) Se usa
Task.Run
para llamar a métodos vinculados a la CPU.Debe usar
Task.Run
, pero no dentro de ningún código que desee que sea reutilizable (es decir, código de biblioteca). Por lo tanto se utilizaTask.Run
para llamar al método, no como parte de la aplicación del método.Así, el trabajo puramente vinculado a la CPU se vería así:
Que llamarías usando
Task.Run
:Los métodos que son una combinación de CPU y E / S deben tener una
Async
firma con documentación que indique su naturaleza de CPU:A lo que también llamarías usando
Task.Run
(ya que está parcialmente vinculado a la CPU):fuente
ConfigureAwait(false)
. Si lo haces primero, entonces puedes encontrar queTask.Run
es completamente innecesario. Si no todavía necesitaTask.Run
, entonces no hay mucha diferencia con el tiempo de ejecución en este caso si se llama a una o varias veces, por lo que acaba de hacer lo que es más natural para su código.ConfigureAwait(false)
en su método enlazado a la CPU, sigue siendo el hilo de la interfaz de usuario que va a hacer el método enlazado a la CPU, y solo todo lo que se pueda hacer después se puede hacer en un hilo TP. ¿O entendí mal algo?Task.Run
entiende las firmas asincrónicas, por lo que no se completará hasta queDoWorkAsync
se complete. El extraasync
/await
es innecesario. Explico más sobre el "por qué" en mi serie de blogs sobreTask.Run
etiqueta .TaskCompletionSource<T>
una de sus anotaciones abreviadas comoFromAsync
. Tengo una publicación de blog que detalla más por qué los métodos asincrónicos no requieren hilos .Un problema con su ContentLoader es que internamente funciona secuencialmente. Un mejor patrón es paralelizar el trabajo y luego sincronizarlo al final, para obtener
Obviamente, esto no funciona si alguna de las tareas requiere datos de otras tareas anteriores, pero debería brindarle un mejor rendimiento general para la mayoría de los escenarios.
fuente