Back to questions
Middle+
3
// При обновлении заказа, мы должны отправить данные по заказу в несколько сервисов (сторонние API) // количество сервисов растёт (может быть тысяч+) // мы написали код, сначала все было хорошо, но со временем наш сервис начал потреблять много памяти func (s *orderService) SendOrder(ctx context.Context, hosts []string, order Order) { for i := 0; i < len(hosts); i++ { go func() { // Представим, что это долгий сетевой вызов response, err := s.httpClient.Send(ctx, hosts[i], order) if err != nil { s.logger.Error(ctx, "failed to send", err) return } s.logger.Info(ctx, "success", response) }() } }
Companies where asked
Lenta Tech
Answer from AI
sobes.tech AI
В представленном коде есть проблема с использованием переменной цикла внутри горутины — переменная hosts[i] может измениться к моменту выполнения горутины, что приведёт к неправильным вызовам.
Кроме того, при большом количестве сервисов (тысячи и более) запуск тысяч горутин одновременно приведёт к высокому потреблению памяти и ресурсов.
Рекомендации по исправлению:
- Передавать переменную в замыкание, чтобы избежать захвата изменяющейся переменной:
for i := 0; i < len(hosts); i++ {
host := hosts[i]
go func() {
response, err := s.httpClient.Send(ctx, host, order)
if err != nil {
s.logger.Error(ctx, "failed to send", err)
return
}
s.logger.Info(ctx, "success", response)
}()
}
- Ограничить количество одновременно работающих горутин с помощью семафора или worker pool, чтобы не создавать тысячи горутин одновременно:
sem := make(chan struct{}, 100) // ограничение до 100 параллельных вызовов
for _, host := range hosts {
sem <- struct{}{}
go func(h string) {
defer func() { <-sem }()
response, err := s.httpClient.Send(ctx, h, order)
if err != nil {
s.logger.Error(ctx, "failed to send", err)
return
}
s.logger.Info(ctx, "success", response)
}(host)
}
// дождаться завершения всех горутин
for i := 0; i < cap(sem); i++ {
sem <- struct{}{}
}
Таким образом, мы избегаем утечек памяти и контролируем нагрузку на систему.