Optimiza la eficiencia de tu código concurrente con DiscardingTaskGroup

En ocasiones, no sabemos la cantidad de procesos concurrentes que necesitaremos en tiempo de desarrollo, en eso estos utilizamos TaskGroup. Ya hablamos inicialmente sobre estos en un artículo anterior.

Antes, cuando necesitábamos manejar procesos concurrentes sin retornar nada, teníamos que usar TaskGroup y especificar un tipo, como Void.self. Veamos un ejemplo:

func execute() async {
await withTaskGroup(of: Void.self) { group in
for _ in list {
group.addTask {
await fetch()
}
}
...
}
}

En este código recorremos un arreglo list y creamos una tarea para ejecutar fetch() por cada miembro del arreglo.

Pero ahora, con la nueva incorporación en WWDC 2023, contamos con la API WithDiscardingTaskGroup, lo que nos permite simplificar el código:

func execute() async {
await withDiscardingTaskGroup { group in
for _ in list {
group.addTask {
await fetch()
}
}
...
}
}

¡Genial!, ¿Verdad? Ya no tenemos que preocuparnos por especificar un tipo de retorno como Void.self.

Beneficios de usar DiscardingTaskGroup

Usar WithDiscardingTaskGroup trae consigo otros beneficios que nos ayudarán a escribir código más eficiente:

  • No retienen los resultados: los recursos utilizados por las tareas hijas se liberan inmediatamente después de que la tarea finaliza. Esto ayuda a reducir el consumo de memoria cuando se tienen muchas tareas que no necesitan devolver nada.
  • Eliminan automáticamente a sus tareas hijas: por lo que no es necesario cancelar explícitamente el grupo o hacer limpieza. Ya no necesitamos usar group.cancelAll(), como cuando usábamos withThrowingTaskGroup.
  • Manejo de errores: si alguna de las tareas hijas genera un error, todas las tareas restantes se cancelan automáticamente. Para cuando tengamos grupos de tareas que puedan generar un error, contamos con withThrowingDiscardingTaskGroup.

Convirtiendo TaskGroup en DiscardingTaskGroup

Un código como el siguiente, donde debemos:

  • Ejecutar try await group.next() para poder ir almancenado los valores retornados.
  • Ejecutar group.cancelAll() para liberar recursos.
try? await withThrowingTaskGroup(of: Void.self) { group in
for _ in list {
group.addTask {
await fetch()
}
}
group.addTask {
try await Task.sleep(for: shiftDuration)
}
try await group.next()
group.cancelAll()
}

Se convertirá en lo siguiente si usamos WithDiscardingTaskGroup:

try? await withThrowingDiscardingTaskGroup { group in
for _ in list {
group.addTask {
await fetch()
}
}
group.addTask {
try await Task.sleep(nanoseconds: 1_000)
}
}

Con este pequeño cambio aprovecharemos todos los beneficios que nos ofrece withThrowingDiscardingTaskGroup.

Comparte este artículo

Subscríbete a nuestro Newsletter

Mantente al día en el mundo de las aplicaciones móviles con nuestro blog especializado.

Artículos semanales

Todas las semanas artículos nuevos sobre el mundo de las aplicaciones móviles.

No spam

No te enviaremos spam, solo contenido de calidad. Puedes darte de baja cuando quieras.

Contenido de calidad

Nada de contenido generado de manera automática usando ChatGPT.

Recomendaciones

Tips indispensables sobre mejores prácticas y metodologías.

© 2024 AsyncLearn