Utiliza este crédito para lanzar tus proyectos ahora en Digital Ocean, válido por 60 días
Índice del contenido
Go: channels, entendiendo los deadlocks o puntos muertos
Go: channels, entendiendo los deadlocks o puntos muertos
Cuando trabajamos con channels hay un error bastante común que ocurre cuando no se está familiarizado con los conceptos, el error es “fatal error: all goroutines are asleep - deadlock!”, traducido significa algo como “error fatal: todas las goroutines están dormidas - en un punto muerto”. La primera vez que vi este error me confundió mucho y, aunque sabía como solucionarlo, no entendía porque sucedía, así que en esta entrada te explico porque sucede como me hubiera gustado haberlo leído en su momento.
¿Por qué sucede el error fatal error: all goroutines are asleep - deadlock!?
Este error sucede cuando:
- Un canal manda información, pero no uno que la reciba.
- Existe un canal que recibe información, pero no uno que la mande.
- Cuando no estamos dentro de una goroutine diferente a la de la función main
En cualquiera de estos casos las goroutines se quedan “esperando” ya sea mandar o recibir información, por lo que se puede decir que estamos “atascados” y es entonces cuando recibimos el error fatal error: all goroutines are asleep - deadlock!
¿Por qué no ir más allá de los fundamentos de Go hacia algo más avanzado?
PatrocinadoEl otro día encontré este curso de Go, trata sobre el nucleo de Go como lenguaje, así como conceptos avanzados como manejo de errores y redes, técnicas de programación eficientes y errores comunes de este incomprendido lenguaje tan potente.
Operaciones bloqueantes
En go, las operaciones que mandan o reciben valores de canales son bloqueantes dentro de su propia goroutine (recuerda que la función main es una goroutine), es decir, mantienen la ejecución del código en espera :
-
Si una operación recibe información de un canal, se bloqueará hasta que la reciba.
-
Si una operación manda información a un canal, se bloqueará hasta que la información enviada sea recibida.
Conociendo las dos situaciones anteriores tendremos dos casos principales:
Bloqueo o deadlock por falta de remitente
¿Qué pasa si una operación está esperando recibir información de un canal, pero ese canal no va a mandar nada jamás?
Sucede cuando existe un canal que recibe información, pero no uno que la mande.
package main
import (
"fmt"
)
func main() {
c := make(chan string)
fmt.Println(<-c)
}
// fatal error: all goroutines are asleep - deadlock!
Bloqueo o deadlock por falta de destinatario
¿Y si una operación manda información a un canal pero ya no hay ninguna otra goroutine que reciba la información de ese canal?
Sucede cuando existe un canal que manda información, pero no uno que la reciba.
package main
func main() {
c := make(chan string)
c <- "texto"
}
// fatal error: all goroutines are asleep - deadlock!
Bloqueo por ausencia de goroutine
Existe un tercer caso: si hay una operación que escribe y lea de un canal, como en el ejemplo de abajo, pero no se encuentra dentro de otra goroutine (diferente a la goroutine de main) que lea el valor, obtendremos un error de deadlock o bloqueo.
package main
import "fmt"
func main() {
var channel = make(chan string)
channel <- "forty two"
fmt.Println(<-channel) // Esto debería de estar en otra goroutine
fmt.Println("Finished")
}
// fatal error: all goroutines are asleep - deadlock!
En los tres casos anteriores el programa se quedaría esperando y, como ya sabes, no tiene sentido mantener el programa esperando por algo que jamás sucederá, por lo que la ejecución se corta y se genera el error fatal: “fatal error: all goroutines are asleep - deadlock!”
A la situación anterior se le conoce como un deadlock o punto muerto.
Deadlocks o puntos muertos
En un deadlock hay una goroutine que está esperando leer o escribir en un canal, sin embargo ya no hay ninguna goroutine ejecutándose pues están esperándose las unas a las otras; están en un punto muerto del que no se puede avanzar.
Prevenir deadlocks en go con goroutines
En la introducción a los channels o canales en go , te dije que la capacidad por defecto de un canal es de 0, esto provoca que no podamos almacenar datos en los canales de manera predeterminada. Si intentamos almacenar un dato en un canal, obtendremos un error por parte del compilador, pues ya no existe otra goroutine que reciba el valor de manera inmediata.
Para prevenir un deadlock, podemos usar inmediatamente el dato del canal creando una goroutine que use el valor del canal.
package main
import "fmt"
func main() {
var channel = make(chan int)
go func(channel chan int) {
channel <- 42
}(channel)
fmt.Println(<-channel)
fmt.Println("Finished")
}
// No ocurre el error de deadlock
En este caso he creado una goroutine con una función anónima que se encarga de leer del canal.
Prevenir deadlocks en go con buffered channels
Si un canal no tiene buffer, el valor queda “retenido” hasta que se reciba, bloqueando la ejecución mientras sucede, ocasionando el deadlock o punto muerto si nadie lo recibe.
Por otro lado, si el canal tiene buffer, entonces la ejecución se bloquea hasta que el valor se haya copiado al buffer, por lo que no obtendremos un error, incluso si ninguna goroutine lo recibe.
package main
import "fmt"
func main() {
var channel = make(chan int, 1)
channel <- 42
fmt.Println(<-channel)
fmt.Println("Finished")
}
// No ocurre el error de deadlock
Recursos sobre deadlocks en go
Para concluir el artículo, te comparto algunos recursos interesantes que hablan de los deadlocks o puntos muertos.