ホームページ >バックエンド開発 >Golang >応答を返した後に Go HTTP サーバーを正常にシャットダウンするにはどうすればよいですか?

応答を返した後に Go HTTP サーバーを正常にシャットダウンするにはどうすればよいですか?

DDD
DDDオリジナル
2024-10-28 19:12:29220ブラウズ

How to Gracefully Shut Down a Go HTTP Server After Returning a Response?

応答を返した後に HTTP サーバーをシャットダウンする

このシナリオでは、Instagram API と対話する Go コマンドライン ボットを構築しました。アクセス トークンを取得するには、ローカル HTTP サーバーを起動してリダイレクト URI を取得します。ただし、アクセス トークンを取得した後にサーバーを手動でシャットダウンしようとすると、問題が発生します。

問題:

シャットダウン時に「サーバーが閉じられました」および「無効なメモリ アドレス」エラーが発生するサーバー

分析:

発生したエラーは、すべての接続が正常に閉じられる前に HTTP サーバーが途中でシャットダウンされていることを示しています。これは、HTTP ハンドラーとサーバーのシャットダウン プロセスの非同期動作が原因で発生する可能性があります。

解決策 1: Context.WithCancel:

context.WithCancel を使用して、キャンセル可能なコンテキストを作成することを検討してください。アクセストークンが取得されたとき。このコンテキストを HTTP ハンドラーに渡し、CancelFunc を使用して、コンテキストが閉じられたときにサーバーを正常にシャットダウンします。

<code class="go">package main

import (
    "context"
    "io"
    "log"
    "net/http"
)

func main() {
    ctx, cancel := context.WithCancel(context.Background())

    http.HandleFunc("/quit", func(w http.ResponseWriter, r *http.Request) {
        io.WriteString(w, "Bye\n")
        cancel()
    })
    http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
        io.WriteString(w, "Hi\n")
    })

    srv := &http.Server{Addr: ":8080"}
    go func() {
        err := srv.ListenAndServe()
        if err != http.ErrServerClosed {
            log.Println(err)
        }
    }()

    <-ctx.Done()

    err := srv.Shutdown(context.Background())
    if err != nil {
        log.Println(err)
    }

    log.Println("done.")
}</code>

解決策 2: 関数間で同じコンテキストを使用する:

を使用することもできます。複数の HTTP ハンドラーと main 関数にわたって同じコンテキストを使用して、サーバーをシャットダウンする必要があるときにすべての関数に通知されるようにします。

<code class="go">package main

import (
    "context"
    "io"
    "log"
    "net/http"
)

func main() {
    ctx, cancel := context.WithCancel(context.Background())
    http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
        io.WriteString(w, "Hi\n")
    })
    http.HandleFunc("/quit", func(w http.ResponseWriter, r *http.Request) {
        io.WriteString(w, "Bye\n")
        cancel()
    })
    srv := &http.Server{Addr: ":8080"}
    go func() {
        if err := srv.ListenAndServe(); err != nil {
            log.Printf("Httpserver: ListenAndServe() error: %s", err)
        }
    }()
    <-ctx.Done()

    if err := srv.Shutdown(ctx); err != nil && err != context.Canceled {
        log.Println(err)
    }
    log.Println("done.")
}</code>

結論:

context.WithCancel を使用するか、複数の機能にわたって同じコンテキストを使用できるため、すべての接続が閉じられた後に HTTP サーバーが正常にシャットダウンすることが保証されます。これにより、現在発生しているエラーが回避されます。

以上が応答を返した後に Go HTTP サーバーを正常にシャットダウンするにはどうすればよいですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。