如何正确关闭请求并继续在后台处理它

时间:2016-08-29 09:27:20

标签: http go

对于传入的HTTP请求,我必须使用202 Accepted状态代码进行响应,同时继续在后台处理有效负载。例如,这就是我目前正在做的事情:

package main

import (
    "fmt"
    "log"
    "net/http"
    "time"

    "github.com/nbari/violetear"
)

func sleep() {
    time.Sleep(3 * time.Second)
    fmt.Println("done...")
}

func index(w http.ResponseWriter, r *http.Request) {
    w.WriteHeader(http.StatusAccepted)
    go sleep()
}

func main() {
    router := violetear.New()
    router.HandleFunc("*", index)

    http.Handle("/", router)
    log.Fatal(http.ListenAndServe(":8080", router))
}

基本上,在处理程序上我只使用WriteHeader,稍后调用goroutine中的sleep函数:

func index(w http.ResponseWriter, r *http.Request) {
    w.WriteHeader(http.StatusAccepted)
    go sleep()
}

如果我想回复" 200 OK",我注意到我可以简单回复,例如:

func index(w http.ResponseWriter, r *http.Request) {
    go sleep()
    return
}

因此想知道我是否应该总是回来:

func index(w http.ResponseWriter, r *http.Request) {
    w.WriteHeader(http.StatusAccepted)
    go sleep()
    return
}

或者只是编写标题然后调用goroutine即可。

1 个答案:

答案 0 :(得分:2)

从处理程序返回就足够了,应该做什么。引自http.Handler

  

返回请求已完成的信号;在完成ServeHTTP调用之后或同时使用ResponseWriter或从Request.Body读取是无效的。

请注意,最后的return语句不是必需的,您可以简单地省略它。当执行最后一条语句时执行从处理程序返回,执行不等待从函数启动的goroutines完成。 (请注意,deferred语句将在之前执行,但您在此处没有。)

同样在返回时,如果未设置HTTP标头,则会自动设置200 OK。因此,如果您需要202 Accepted,则以下是最低要求:

func index(w http.ResponseWriter, r *http.Request) {
    w.WriteHeader(http.StatusAccepted)
    go sleep()
}

请确保在从处理程序返回后不会在并发goroutine中使用http.ResponseWriterhttpRequest值,因为它们可能会被重用,因此您甚至不应尝试读取它们。