Concorrência em Go

Em Go é possível fazer programas que utilizam de concorrência para melhorar o seu desempenho. A concorrência consiste em executar múltiplas funções independentes ao mesmo tempo para tentar otimizar a utilização de recursos e diminuir o tempo gasto com processamento.

Suponhamos que temos uma API que faz duas requisições para APIs de terceiros e cada uma delas demore 1 segundo. Em um código sequencial sem concorrência, nossa API demoraria no mínimo 2 segundos para responder. Caso as requisições para essas duas APIs sejam independentes, é possível fazer essas chamadas ao mesmo tempo. Essa abordagem poderia diminuir em 50% o tempo de resposta da nossa API somente com essa simples alteração para utilizar concorrência.

Sem concorrência

Abaixo vemos a imagem que representa o fluxo sem concorrência onde as requisições para serviços externos são feitas em sequência.

Código sem concorrência

package main

import (
	"log"
	"time"
)

func main() {
	log.Println("inicio")
	log.Println(requestApiTerceiros1())
	log.Println(requestApiTerceiros2())
	log.Println("fim")
}

func requestApiTerceiros1() string {
	time.Sleep(1 * time.Second)
	return "resposta api terceiros 1"
}

func requestApiTerceiros2() string {
	time.Sleep(1 * time.Second)
	return "resposta api terceiros 2"
}

Resultado da execução do código

go run main.go
2021/10/11 16:59:35 inicio
2021/10/11 16:59:36 resposta api terceiros 1
2021/10/11 16:59:37 resposta api terceiros 2
2021/10/11 16:59:37 fim

Com concorrência

Ao utilizar concorrência ambas as requisições podem ser feitas ao mesmo tempo para diminuir o tempo como mostra o fluxo da imagem abaixo.

Código com concorrência

package main

import (
	"log"
	"time"
)

func main() {
	log.Println("inicio")
	go requestApiTerceiros1()
	go requestApiTerceiros2()
	time.Sleep(5 * time.Second)
}

func requestApiTerceiros1() {
	time.Sleep(1 * time.Second)
	log.Println("resposta api terceiros 1")
}

func requestApiTerceiros2() {
	time.Sleep(1 * time.Second)
	log.Println("resposta api terceiros 2")
}

Resultado da execução do código

go run main2.go
2021/10/11 17:01:17 inicio
2021/10/11 17:01:18 resposta api terceiros 2
2021/10/11 17:01:18 resposta api terceiros 1

Em suma, podemos ver que conseguimos otimizar nosso código usando concorrência. Desse modo, é importante saber como trabalhar com concorrência para ter a habilidade de fazer o uso dessa característica da linguagem quando for apropriado.

É importante conhecer alguns conceitos da linguagem para trabalhar bem com concorrência, são eles:

  • Go Routines
  • Channels
  • WaitGroup
  • Select

Go Routines

Uma Go routine é uma thread leve que é gerenciada pelo Go e é utilizada para rodar algum código específico que esteja dentro de uma função.

Não é preciso ter nenhuma característica específica para conseguir executar uma função dentro de uma Go Routine, basta utilizar a palavra chamada go antes da chamada dessa função.

Chamada de função normal

func main(){
	sayHi(1)
}

func sayHi(n int){
	fmt.Printf("hi %d\n", n)
}

Chamada de função com Go Routine

func main(){
	go sayHi(1)
}

Ao executar ambos os códigos é possível notar que na chamada normal de função é exibido na tela o texto normalmente, enquanto que ao chamar utilizando Go Routine o texto não é exibido.

O motivo de isso acontecer é porque esse código executa a função sayHi em uma thread separada e não fica esperando a finalização dessa thread. Como não existe mais nenhum código após a criação dessa Go Routine então o programa é finalizado sem nunca ter o retorno da função. Um jeito simples de resolver isso é adicionar um tempo de espera no código da seguinte forma:

func main(){
	go sayHi(1)
	time.Sleep(1 * time.Second)
}

Apesar dessa abordagem resolver o problema, ela está longe de ser a ideal pois nunca saberemos exatamente o tempo necessário para uma função ser executada e dessa forma estamos adicionando atraso no nosso código.

Esse exemplo acima foi feito apenas para entender a sintaxe utilizada para executar funções dentro de Go Routines e mostrar que o código não aguarda a execução dessas funções antes de continuar seu fluxo normal. Para o gerenciamento correto de Go Routines o Go possui mecanismos próprios para isso como Channel e WaitGroup.

Channels

Channel é uma maneira que o Go possui de permitir a comunicação entre diferentes partes do código. Um dos cenários possíveis para se utilizar channels é para a comunicação entre Go Routines.

Para criar um novo channel basta criar uma nova variável do tipo chan, por exemplo:

var out chan string // valor default é nil
out := make(chan string, 1) // cria um channel vazio do tipo string que é capaz de armazenar 1 elemento

Ao criar uma variável do tipo channel o seu valor default é nil. Para inicializar o channel como sendo um channel vazio é preciso utilizar a função make, assim como é feito com variáveis do tipo slice e map.

O operador utilizado para inserir ou ler dados de um channel é o operador <-, com o sentido dessa seta apontando para onde o dado está indo.

Código

out := make(chan string, 1)
out <- "myString" //Adiciona uma string no channel criado
fmt.Println(<-out) //Imprime string armazenada no channel criado

Resultado

myString

Um fato interessante sobre como channels funcionam é que ao tentar ler o conteúdo de um channel vazio ou ao tentar escrever um dado em um channel sem espaço o código pára e fica esperando até ter espaço suficiente no channel. Podemos ver esse comportamento ao usar o exemplo acima só que removendo a escrita no channel:

Código

out := make(chan string, 1)
fmt.Println(<-out) //Imprime string armazenada no channel criado

Resultado

fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan receive]:
main.main()
        /home/breno/programming/tmp/main.go:10 +0x39
exit status 2

A mensagem de erro mostra que ocorreu um deadlock pois não existe nenhuma Go Routine sendo executada além da função main e ela está parada esperando algo ser escrito no channel. O deadlock ocorre porque nunca será escrito algo no channel já que o main está parado tentando ler do canal e não existe nenhuma outra Go Routine sendo executada. Se criarmos uma Go Routine que não faz nada e que fique rodando direto, o erro não ocorre mais. Veja o exemplo abaixo: Código

func main() {
	out := make(chan string, 1)
	go func() {
		for {

		}
	}()
	fmt.Print(<-out)
}

Resultado

Nada impresso na tela

Se for parar pra pensar, essa abordagem para solucionar o erro de deadlock nesse exemplo não resolve nenhum problema de fato pois esse problema de deadlock só existe porque estamos usando channel em um local que deveríamos estar usando uma simples variável do tipo string. No entanto, haverá casos onde essa tática será útil como quando formos construir alguma pipeline para execução de código concorrente.

Compartilhando channels entre Go Routines

É possível compartilhar um channel entre várias Go Routines que estejam sendo executadas ao mesmo tempo como mostra o exemplo abaixo. Nesse código quatro diferentes rotinas são executadas ao mesmo tempo e a medida que dados são escritos no channel eles são mostrados na tela.

func main() {
	out := make(chan string)
	go sayHi(1, out)
	go sayHi(2, out)
	go sayHi(3, out)
	go sayHi(4, out)

	for i := 1; i <= 4; i++ {
		fmt.Println(<-out)
	}
}

func sayHi(n int, out chan string) {
	out <- fmt.Sprintf("hi %d", n)
}

É interessante notar que o mesmo channel está sendo usado nas quatro Go Routines criadas. O channel foi criado com a função make sem passar o tamanho como segundo argumento, isso indica que o channel criado não possui buffer. Na prática isso indica que um dado só poderá fluir nesse canal quando tiver uma parte do código lendo e outra escrevendo pois não existe nenhum local para armazenar esse valor.

No exemplo acima, assim que uma Go Routine escrever no canal o channel será bloqueado para escrita até esse valor for removido do canal pela função Println, sendo assim não existe uma concorrência de fato nesse código já que as Go Routines criadas estão sempre aguardando a liberação do canal.

Também nesse exemplo, o programa não garante ordem de execução das Go Routines visto que são threads de execução independentes.

WaitGroup

Um WaitGroup permite parar a execução do código e aguardar até que todas as Go Routines escolhidas terminem sua execução. Dessa forma, é possível fazer código que depende do resultado de múltiplas Go Routines. Veja abaixo um exemplo de uso de WaitGroup para aguardar a execução da execução de 4 Go Routines.

package main

import (
	"fmt"
	"sync"
)

func main() {
	// Cria uma variável do tipo WaitGroup
	var wg sync.WaitGroup

	for i := 1; i < 5; i++ {
		// Adiciona 1 ao contador de Go Routines do WaitGroup wg
		wg.Add(1)

		// Cria Go Routine passando o WaitGroup criado como parâmetro
		go sayHi(&wg, i)
	}

	// Aguarda até o contador de Go Routines do WaitGroup wg chegar a zero
	wg.Wait()

	fmt.Println("Fim")
}

func sayHi(wg *sync.WaitGroup, n int) {
	fmt.Println(n)

	// Decrementa o contador do WaitGroup wg
	wg.Done()
}

Select

Select é uma forma de escolher qual dos caminhos o código irá seguir, similar ao funcionamento do switch, mas nesse caso o select é utilizado com Channels. Caso o select não possua um caminho default o código fica parado até que algum dos casos sejam satisfeitos. Veja abaixo um exemplo funcional e outro onde há deadlock.

Código funcional

package main

import "fmt"

func main() {
	var c1, c2 chan int
	select {
	case <-c1:
	case <-c2:
	default:
		fmt.Println("Fim")
	}
}

Código com deadlock

package main

func main() {
	var c1, c2 chan int
	select {
	case <-c1:
	case <-c2:
	}
}

Resultado da execução desse código com deadlock

fatal error: all goroutines are asleep - deadlock!

goroutine 1 [select]:
main.main()
        /home/breno/study-programming/go/select.go:5 +0x45
exit status 2