Какие существуют способы для корректного завершения (graceful shutdown) горутин в Go?

Ответ

В Go нет механизма для принудительного "убийства" горутины извне. Вместо этого используется принцип кооперативной отмены: горутина должна сама проверять сигнал о необходимости завершения и корректно останавливать свою работу.

Существует два основных подхода:

1. Использование канала (done channel)

Это классический способ. Создается отдельный канал, который закрывается, когда нужно дать сигнал горутинам на завершение.

// worker ожидает сигнала в канале done
func worker(done <-chan struct{}) {
    fmt.Println("Worker запущен...")
    select {
    case <-done: // Если канал done закрыт, этот case сработает
        fmt.Println("Worker останавливается.")
        return
    case <-time.After(5 * time.Second): // Пример таймаута
        fmt.Println("Worker завершился по таймауту.")
        return
    }
}

func main() {
    done := make(chan struct{})
    go worker(done)

    // Имитируем работу и даем сигнал на остановку через 2 секунды
    time.Sleep(2 * time.Second)
    close(done) // Закрытие канала рассылает сигнал всем слушающим горутинам
    time.Sleep(1 * time.Second) // Даем время на завершение
}

2. Использование пакета context (предпочтительный способ)

Пакет context — это идиоматичный и более мощный инструмент для управления отменой, таймаутами и передачей request-scoped данных.

// worker ожидает сигнала отмены через контекст
func worker(ctx context.Context) {
    fmt.Println("Worker запущен...")
    select {
    case <-ctx.Done(): // Канал Done() закрывается при отмене контекста
        // ctx.Err() возвращает причину отмены
        fmt.Printf("Worker останавливается: %vn", ctx.Err())
        return
    }
}

func main() {
    // Создаем контекст, который можно отменить
    ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
    defer cancel() // Важно всегда вызывать cancel для освобождения ресурсов

    go worker(ctx)

    // Ждем, пока сработает таймаут (3 секунды)
    time.Sleep(4 * time.Second)
}

Ключевой вывод: Независимо от способа, горутина должна активно слушать сигнал о завершении (через select) и самостоятельно прекращать свою работу. Это предотвращает утечки горутин (goroutine leaks).