ホームページ >バックエンド開発 >Golang >私のラップトップは Go を使用して 100 万件の同時 HTTP リクエストを処理できますか?

私のラップトップは Go を使用して 100 万件の同時 HTTP リクエストを処理できますか?

Linda Hamilton
Linda Hamiltonオリジナル
2024-12-01 17:01:081014ブラウズ

Can My Laptop Handle One Million Concurrent HTTP Requests Using Go?

あなたのラップトップは 100 万件の同時 HTTP リクエストを処理できますか?

1,000,000 件の HTTP リクエストを同時に REST API サービスに送信し、コンピュータの制限を押し上げることを想像してみてください。同時実行性を最大化します。このタスク用のツールは存在しますが、ゴルーチンを使用して Go でこのタスクを実行する方法を詳しく見てみましょう。

package main

import (
    "fmt"
    "net/http"
    "runtime"
    "time"
)

func main() {
    runtime.GOMAXPROCS(runtime.NumCPU()) // Use all available CPU cores
    transport := &http.Transport{}        // Create an HTTP transport

    for i := 0; i < 1000000; i++ {
        go func() { // Start a goroutine for each HTTP request
            req, _ := http.NewRequest("GET", "http://myapi.com", nil)
            req.Header.Set("User-Agent", "custom-agent")
            req.SetBasicAuth("xxx", "xxx")

            resp, err := transport.RoundTrip(req)
            if err != nil {
                panic("HTTP request failed.")
            }
            defer resp.Body.Close()

            if resp.StatusCode != 302 {
                panic("Unexpected response returned.")
            }

            location := resp.Header.Get("Location")
            if location == "" {
                panic("No location header returned.")
            }

            fmt.Println("Location Header Value:", location)
        }()
    }

    time.Sleep(60 * time.Second) // Sleep for 60 seconds
}

しかし、ちょっと待ってください。問題があります!

このスクリプトの実行ファイル記述子の制限によりエラーが発生します。システムは、あまりにも多くの同時接続を処理できません。

改善されたソリューション

これらの制限を克服するには、より洗練されたアプローチを使用する必要があります。

ディスパッチャーとワーカープールの使用

これ解決策には、リクエストをチャネルにプッシュするディスパッチャ ゴルーチンの作成が含まれます。ゴルーチンのワーカー プールはチャネルからリクエストを取得し、処理して、応答チャネルに送信します。その後、コンシューマのゴルーチンが応答を処理します。

// Dispatcher function
func dispatcher(reqChan chan *http.Request, reqs int) {
    defer close(reqChan)
    for i := 0; i < reqs; i++ {
        req, err := http.NewRequest("GET", "http://localhost/", nil)
        if err != nil {
            log.Println(err)
        }
        reqChan <- req
    }
}

// Worker function
func worker(t *http.Transport, reqChan chan *http.Request, respChan chan Response) {
    for req := range reqChan {
        resp, err := t.RoundTrip(req)
        r := Response{resp, err}
        respChan <- r
    }
}

// Consumer function
func consumer(respChan chan Response, reqs int) (int64, int64) {
    var (
        conns int64
        size  int64
    )
    for conns < int64(reqs) {
        select {
        case r, ok := <-respChan:
            if ok {
                if r.err != nil {
                    log.Println(r.err)
                } else {
                    size += r.ContentLength
                    if err := r.Body.Close(); err != nil {
                        log.Println(r.err)
                    }
                }
                conns++
            }
        }
    }
    return conns, size
}

結果

この改良されたスクリプトを実行すると、次のような印象的な結果が生成されます。

接続: 1000000

同時実行数: 200

合計サイズ: 15000000 バイト

合計時間: 38m20.3012317s

平均時間: 2.280131ms


パフォーマンスの最適化

ただし、同時リクエスト数と合計リクエスト数を調整すると、パフォーマンスを向上させるのに役立ちますシステムの制限とその機能のストレス テストを行います。これは極端なテストであり、システム リソースを急速に消費する可能性があることに注意してください。

以上が私のラップトップは Go を使用して 100 万件の同時 HTTP リクエストを処理できますか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

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