Entendiendo async y await en Javascript

Entendiendo async y await en Javascript

Async/Await

En un artículo anterior revisamos que son las promesas y como trabajar con ellas, a modo de recordatorio:

const sleep = time => new Promise((resolve, reject) => setTimeout(resolve, time));

sleep(3000).then(() => {  
    console.log('¿Funcionó la promesa?')  
})

Esta función sleep retorna una promesa, para ejecutar "algo" cuando la promesa es resuelta utilizamos then y pasamos una función.

En modo general puedes considerar una función que retorna una promesa como una función asíncrona. Pero, esto no es efectivamente declarado en el código, si no más bien inferido dado que la función retorna una promesa. ¿Qué tal si tuvieras una manera de declarar que una promesa es asíncrona?.

Funciones asíncronas.

Como siempre, encuentra el demo de este artículo aquí

Una función asíncrona es aquella declarada utilizando la palabra clave async. Esta puede ser usada en funciones normales o en arrow functions.

async function doSomething() {  
}

const doSomething = async () => {  
}

¿Qué significa esta declaración async?

Básicamente lo que estás haciendo es indicándole al intérprete que la función doSomething retornará siempre una promesa.

const doSomething = async () => {  
    return 1  
}

// es lo mismo

const doSomething = () => new Promise(resolve => resolve(1))

// ambas se usan igual

const promise = doSomething()  
promise.then(result => {  
    console.log(reuslt)  
})

Pero, a pesar de ser semánticamente lo mismo para el interprete el uso de async nos ofrece una nueva característica

await

El uso de async para declarar tu función te permite usar una nueva palabra clave: await que de forma muy resumida, te permite literalmente esperar a que la promesa esté resuelta.

// sin async/await
const getData = () => {
  console.log('Fetching...')
  fetchData().then(result => {
    // hacer algo con el resultado
    console.log(result)
    return result
  })
}

// usando async/await

const asyncGetData = async () => {
  console.log('Fetching async....')
  const result = await fetchData()
  console.log(result)
}

En el ejemplo tenemos la misma implementación, primer con promesa y luego con async/await la diferencia es clara, el uso de await permite una mejor legibilidad. Básicamente, await es un reemplazo del código que escribes dentro de la llamada a then en una promesa y te permite escribir código de forma síncrona.

Pero, tu bien sabes que una promesa puede ser resuelta o rechazada y notaste que await solo determina el camino feliz de una promesa resuelta con éxito. ¿Que ocurre con los errores?.

Manejo de errores

El objeto Promise nos ofrece 3 métodos que te permiten controlar el flujo de la promesa o “suscribirte a los cambios de estado”, estos son then, catch y finally.

then se ejecuta en caso de una promesa exitosa y es lo que conseguimos al utilizar await.

catch define el trozo de código que se ejecutará si la promesa fue rechazada, es decir, tu manejo de errores. En el caso de async/await no tenemos la misma definición pero, podemos decantar al nunca bien ponderado bloque `try

catch define el trozo de código que se ejecutará si la promesa fue rechazada, es decir, tu manejo de errores. En el caso de async/await no tenemos la misma definición pero, podemos decantar al nunca bien ponderado bloque try/catch

const asyncGetDataWithTry = async () => {
  console.log('Fetching async....')
  try {
    const result = await fetchData()
    console.log('Async Data: ', result)
  }catch(e) {
    console.log(e)
  }  
}

El proceso completo de la promesa se ejecuta dentro del bloque try y si esta promesa llegase a falla, el bloque catch se encarga de ella.

El bloque finally es lo que puedes escribir directamente fuera del bloque try/catch.

Ahora, ¿qué ocurre si tienes varias promesas y varios bloque try/catch ?

const test = async () => {
  try {
    const one = await getOne(false)
  } catch (error) {
    console.log(error)
  }

  try {
    const two = await getTwo(true)
  } catch (error) {
    console.log(error)
  }

  try {
    const three = await getThree(false)
  } catch (error) {
    console.log(error)
  }
}

Eso no se ve nada bien.

¿Qué puedes hacer?

Una forma de mejorar la legibilidad de esto es simplemente poner todo dentro de un sólo bloque try/catch

const test = async () => {
  try {
    const one = await getOne(false)
    const two = await getTwo(false)
    const three = await getThree(false)
  }catch (error) {
    console.log(error) // Failure!
  }
}

idealmente el objeto error retornado por las promesas tiene algún código identificador lo que te permite identificar cual fue el error.

¿De que otra forma puedes escribir el manejo de errores de múltiples funciones asíncronas?
Cuéntame tu solución respondiendo a este email o en un tweet 😎

Conclusión

En resumen async/await es una notación o sintaxis que busca simplificar el proceso de escribir (y sobre todo leer y mantener) código asíncrono transformado el uso de promesas y callbacks en código que se puede seguir de forma síncrona, para esto await espera por el resultado de la promesa deteniendo la ejecución hasta que la promesa se resuelva - el código justo después de await no se ejecuta hasta que la promesa esté lista - .

No olvides que no puedes usar await si no has declarado la función como asíncrona utilizando async.

Desafío

Imagina que tienes que ejecutar múltiples llamadas asíncronas a una API externa, cada llamada a la API demora casi 1 segundo.

const test = async _ => {
  const one = await fetchOne()
  console.log(one)

  const two = await fetchTwo()
  console.log(two)

  const three = await fetchThree()
  console.log(three)

  console.log('Done')
}

Este código resuelve el problema pero crea otro: Cada llamada espera que la anterior termine, y si cada llamada demora 1 segundo, entonces el proceso estará listo solo después de 3 segundos!!.

Y por lo que este código muestra, las llamadas no son dependientes entre si.
¿Cómo puedes escribir este código pero ejecutando las 3 llamadas en paralelo?

Pista: El objeto Promise tiene un método que te permitirá resolver este problema.