65

这个简单的 HTTP 服务器包含对 time.Sleep() 的调用,它使每个请求需要五秒钟。当我尝试在浏览器中快速加载多个选项卡时,很明显每个请求都按顺序排队和处理。我怎样才能让它处理并发请求?

package main

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

func serve(w http.ResponseWriter, r *http.Request) {
   fmt.Fprintln(w, "Hello, world.")
   time.Sleep(5 * time.Second)
}

func main() {
   http.HandleFunc("/", serve)
   http.ListenAndServe(":1234", nil) 
}

其实我是在写完这个问题后才找到答案的,而且很微妙。无论如何我都会发布它,因为我在谷歌上找不到答案。你能看到我做错了什么吗?

4

3 回答 3

110

您的程序已经同时处理请求。您可以ab使用 Apache 2 附带的基准测试工具来测试它:

ab -c 500 -n 500 http://localhost:1234/

在我的系统上,基准测试总共需要 5043 毫秒来处理所有 500 个并发请求。只是您的浏览器限制了每个网站的连接数。

顺便说一句,对 Go 程序进行基准测试并不是那么容易,因为您需要确保您的基准测试工具不是瓶颈,并且它还能够处理那么多并发连接。因此,最好使用几台专用计算机来生成负载。

于 2012-05-23T15:15:53.300 回答
13

从 Server.go ,当连接被接受时, go 例程在 Serve 函数中生成。以下是片段,:-

// Serve accepts incoming connections on the Listener l, creating a 
// new service goroutine for each.  The service goroutines read requests and
// then call srv.Handler to reply to them.
func (srv *Server) Serve(l net.Listener) error {
        for {
            rw, e := l.Accept()
               if e != nil {
......
            c, err := srv.newConn(rw)
            if err != nil {
                continue
            }
            c.setState(c.rwc, StateNew) // before Serve can return
            go c.serve()
        }
}
于 2015-11-24T09:05:44.657 回答
-6

如果您使用 xhr 请求,请确保 xhr 实例是局部变量。

例如,xhr = new XMLHttpRequest()是一个全局变量。当您使用相同的 xhr 变量执行并行请求时,您只会收到一个结果。所以,你必须像这样在本地声明 xhr var xhr = new XMLHttpRequest()

于 2016-06-15T22:22:02.010 回答