Я пытаюсь изящно завершить работу, когда пользователь нажимает Ctrl-C
.Я пытаюсь код в Сделайте Ctrl + C, чтобы отменить контекст. Context .основной пакет
import (
"context"
"fmt"
"os"
"os/signal"
"time"
)
func main() {
ctx := context.Background()
// trap Ctrl+C and call cancel on the context
ctx, cancel := context.WithCancel(ctx)
c := make(chan os.Signal, 1)
signal.Notify(c, os.Interrupt)
defer func() {
signal.Stop(c)
cancel()
fmt.Println("Cleaned up")
}()
go func() {
select {
case <-c:
fmt.Println("Got interrupt signal")
cancel()
case <-ctx.Done():
}
fmt.Println("Stopped monitoring")
}()
select {
case <-ctx.Done():
fmt.Println("notified to quit")
case <-time.NewTimer(time.Second * 2).C:
fmt.Println("done something")
}
}
Работает хорошо, как и ожидалось, когда пользователь нажимает Ctrl-c
, выводит следующее:
Got interrupt signal
Stopped monitoring
notified to quit
Cleaned up
Однако, если он нормально завершается, он не работаеткак и ожидалось, как показано ниже:
done something
Cleaned up
Я имею в виду, что это должно распечатать Stopped monitoring
, но не.В функции defer cleanup
она вызывается cancel()
, которая должна вызвать select in monitoring goroutine
, чтобы выйти, но не.
Как решить проблему?