ホームページ >バックエンド開発 >Golang >Go の HTTP コンテキストがリクエストボディのキャンセルシグナルを検出しないのはなぜですか?

Go の HTTP コンテキストがリクエストボディのキャンセルシグナルを検出しないのはなぜですか?

DDD
DDDオリジナル
2024-12-08 02:43:10565ブラウズ

Why Doesn't Go's HTTP Context Detect Cancellation Signals with Request Bodies?

Go HTTP コンテキストがリクエスト ボディを持つキャンセル シグナルの検出に失敗する

Go の HTTP サーバーでは、コンテキスト シグナルがクライアントの切断を処理するメカニズムを提供できますこれにより、サーバーはリソースを即座にクリーンアップできるようになります。ただし、リクエストに本文が含まれている場合、コンテキストの Done チャネルが停止し、サーバーがクライアントの離脱を検出できなくなります。

動作の原因

この動作の理由HTTP サーバーが接続を読み取る方法にあります。リクエスト本文が読み取られるまで、閉じられた接続のチェックは実行されません。 GET リクエストの場合は本文が存在しないため、サーバーは接続をアクティブに監視します。ただし、POST リクエストでは、本文がノンブロッキング リーダーを占有し、接続チェックが遅れます。

解決策

この問題を解決するには、リクエスト本文を明示的に読み取ります。ハンドラー関数。このアクションにより、サーバーの接続監視がトリガーされ、クライアントの切断を即座に検出できるようになります。

コード例

func handler(w http.ResponseWriter, r *http.Request) {
    go func(done <-chan struct{}) {
        <-done
        fmt.Println("message", "client connection has gone away, request got cancelled")
    }(r.Context().Done())

    // Explicitly read the body to trigger connection monitoring
    io.Copy(ioutil.Discard, r.Body)

    time.Sleep(30 * time.Second)
    fmt.Fprintf(w, "Hi there, I love %s!\n", r.URL.Path[1:])
}

この変更により、サーバーはリクエスト本文を読み取るように求められます。すぐに接続監視プロセスを開始します。その結果、サーバーは、リクエスト本文が存在する場合でも、クライアントの切断を効果的に検出できるようになりました。

以上がGo の HTTP コンテキストがリクエストボディのキャンセルシグナルを検出しないのはなぜですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

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