Вопрос проверяет знание таймеров, каналов и работы runtime Go.
time.After создаёт таймер, который нельзя отменить. Если результат не будет прочитан, таймер останется в памяти до срабатывания.
time.After(d) под капотом создаёт time.Timer и возвращает канал. Если таймер больше не нужен, его невозможно остановить.
Проблема возникает в select:
select {
case <-time.After(time.Minute):
case <-ctx.Done():
}Если ctx.Done() срабатывает первым:
таймер продолжает жить;
канал никто не читает;
при большом количестве вызовов → рост памяти.
Правильный подход — time.NewTimer:
timer := time.NewTimer(time.Minute)
defer timer.Stop()
select {
case <-timer.C:
case <-ctx.Done():
}