[英]why goroutine block main func in this http server?
我想设置一个带有 httprouter 监听两个端口8888
和8080
的 http 服务器,就像下面的代码一样。
package main
import (
"fmt"
"github.com/julienschmidt/httprouter"
"log"
"net/http"
)
func Index(w http.ResponseWriter, r *http.Request, _ httprouter.Params) {
fmt.Fprint(w, "Welcome!\n")
}
func main() {
router := httprouter.New()
router.GET("/", Index)
fmt.Println("listen on 8080")
// this is where blocked
go log.Fatal(http.ListenAndServe(":8080", router))
fmt.Println("listen on 8888")
log.Fatal(http.ListenAndServe(":8888", router))
}
但它不能正常工作,我的服务器只监听8080
。如果我做一些改变:
go func() { log.Fatal(http.ListenAndServe(":8080", router)) }()
它在8080
和8888
上都能很好地工作。为什么? 是closure
还是别的什么?
函数值和参数在调用 goroutine 中照常评估
— Go 语言规范,“Go 语句” 。
您正在创建呼叫到的goroutine log.Fatal
,但参数log.Fatal
预先评估,在主够程。 Fatal
的参数是http.ListenAndServe
的返回值。 所以新的 goroutine 直到ListenAndServe
返回后ListenAndServe
启动。
由于http.ListenAndServe()
是阻塞的,并且在您的场景中,有两个,然后尝试将其中一个放在 goroutine 中。 这个想法是将这两个 Web 服务器初始化语句的执行分开到单独的 goroutine 中。
func main() {
router := httprouter.New()
router.GET("/", Index)
go func() {
fmt.Println("listen on 8080")
log.Fatal(http.ListenAndServe(":8080", router))
}()
fmt.Println("listen on 8888")
log.Fatal(http.ListenAndServe(":8888", router))
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.