Rumah >pembangunan bahagian belakang >Golang >Ralat 'http: menulis melebihi Panjang Kandungan yang diisytiharkan' dalam Go

Ralat 'http: menulis melebihi Panjang Kandungan yang diisytiharkan' dalam Go

WBOY
WBOYke hadapan
2024-02-09 12:50:07774semak imbas

Go 中的“http:写入超过声明的 Content-Length”错误

Editor PHP Xinyi akan memperkenalkan kepada anda hari ini ralat biasa, iaitu ralat "http: menulis melebihi panjang Kandungan yang diisytiharkan" yang berlaku dalam bahasa Go. Apabila kami menulis permintaan HTTP menggunakan bahasa Go, kadangkala kami menghadapi ralat ini. Sebab ralat ini ialah Panjang Kandungan yang kami isytiharkan dalam pengepala permintaan tidak konsisten dengan panjang badan permintaan yang sebenarnya dihantar. Seterusnya, kami akan menerangkan secara terperinci mengapa ralat ini berlaku dan cara menyelesaikannya.

Kandungan soalan

Saya sedang bereksperimen dengan Go dan menulis aplikasi untuk mengurus baris gilir permintaan HTTP yang dikendalikan oleh coroutine pekerja.

Barang concurrency nampaknya berfungsi dengan baik, tetapi selepas dihantar balik http 响应时出现此错误:写入的内容超过声明的 Content-Length.

Kod lengkap adalah seperti berikut:

package main

import (
    "log"
    "net/http"
    "sync"
)

// Job represents a unit of work to be processed by a worker.
type Job struct {
    r *http.Request       // HTTP request to be processed
    w http.ResponseWriter // Response writer to send the result
}

// Queue manages a list of jobs to be processed by workers.
type Queue struct {
    jobs []*Job     // List of jobs in the queue
    mu   sync.Mutex // Mutex to synchronize access to the queue
    cond *sync.Cond // Condition variable for signaling
}

var q Queue // Global instance of the queue
var WORKER_POOL_SIZE = 4 // Number of workers

// Push adds a job to the queue.
func (q *Queue) Push(j *Job) {
    q.mu.Lock()
    defer q.mu.Unlock()
    q.jobs = append(q.jobs, j)
    q.cond.Signal() // Signal a waiting worker that a job is available
    log.Println("Job added to queue")
}

// Pop retrieves and removes a job from the queue.
func (q *Queue) Pop() (*Job, bool) {
    q.mu.Lock()
    defer q.mu.Unlock()
    if len(q.jobs) == 0 {
        q.cond.Wait() // If the queue is empty, wait for a signal
    }
    job := q.jobs[0]
    q.jobs = q.jobs[1:]
    log.Println("Job removed from queue")
    return job, true
}

// handler adds a job to the queue.
func handler(w http.ResponseWriter, r *http.Request) {
    // Create a job with the request and response.
    job := &Job{r, w}

    // Push the job onto the queue.
    q.Push(job)
    log.Println("Received request and added job to queue")
}

// init initializes the condition variable and starts worker goroutines.
func init() {
    q.cond = sync.NewCond(&q.mu)
    for i := 0; i < WORKER_POOL_SIZE; i++ {
        go worker()
    }
}

// worker processes jobs from the queue.
func worker() {
    for {
        job, ok := q.Pop()
        if ok {
            log.Println("Worker processing job")
            doWork(job)
        }
    }
}

// doWork simulates processing a job and sends a response.
func doWork(job *Job) {
    // Extract the "Name" parameter from the request query.
    name := job.r.URL.Query().Get("Name")

    // Check if the name is not empty.
    if name != "" {
        // Send the name as the response.
        _, err := job.w.Write([]byte("Hello, " + name))
        if err != nil {
            log.Println("Error writing response:", err)
        }
        log.Println("Response sent: Hello,", name)
    } else {
        // If the "Name" parameter is missing or empty, send an error response.
        http.Error(job.w, "Name parameter is missing or empty", http.StatusBadRequest)
        log.Println("Error: Name parameter is missing or empty")
    }
}

func main() {
    http.HandleFunc("/addJob", handler)
    log.Println("Server started and listening on :8080")
    err := http.ListenAndServe(":8080", nil)
    if err != nil {
        log.Fatal("Error starting server:", err)
    }
}

Ada idea bagaimana untuk menyelesaikan masalah ini? Selain itu, memandangkan saya sangat baru dalam concurrency in go, adakah anda fikir ia boleh dipertingkatkan? Terima kasih!

Penyelesaian

Mengikut dokumentasi http.Handler:

anda handler 正在将请求/编写器推送到队列中,然后返回。这意味着您在处理程序返回后尝试写入 ResponseWriter ,违反了上述规定(由于没有同步,写入也可能实际发生在 return sebelum atau pada masa yang sama).

Terdapat banyak cara untuk menyelesaikan masalah ini satu teknik ialah:

// Job represents a unit of work to be processed by a worker.
type Job struct {
    r    *http.Request       // HTTP request to be processed
    w    http.ResponseWriter // Response writer to send the result
    done chan struct{}       // Closed when write competed
}

...

func handler(w http.ResponseWriter, r *http.Request) {
    // Create a job with the request and response.
    job := &Job{r, w, make(chan struct{})}

    // Push the job onto the queue.
    q.Push(job)
    log.Println("Received request and added job to queue")
    <-job.done // Wait until job has been processed
}

...

// worker processes jobs from the queue.
func worker() {
    for {
        job, ok := q.Pop()
        if ok {
            log.Println("Worker processing job")
            doWork(job)
            close(job.done) // Notify requester that we are done
        }
    }
}

Ia sangat bergantung pada keperluan. Penyelesaian biasa adalah dengan hanya menggunakan saluran (pengendali menghantar permintaan ke saluran, dan berbilang pekerja menerima dari saluran yang sama).

Atas ialah kandungan terperinci Ralat 'http: menulis melebihi Panjang Kandungan yang diisytiharkan' dalam Go. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Kenyataan:
Artikel ini dikembalikan pada:stackoverflow.com. Jika ada pelanggaran, sila hubungi admin@php.cn Padam