一个最简单的 http 服务器:
func main() {
fs := http.FileServer( http.Dir("./public"))
http.Handle("/", fs)
http.ListenAndServe(":8888", nil)
}
用 curl 测试几个连接同时下载:
curl --limit-rate 1k http://1.2.3.4:9999/file.zip --output NUL
结果是,只有第一个和第二个连接正常下载,其他的连接处于停滞状态,只有前面的连接下载完了,后面的才能接着下载。
如果把服务器换成 nginx 则多个连接都可以同时下载。
以上都是在 Windows 11 下进行的 Go 的版本是 1.18.1
1
shakukansp 2022-06-10 11:17:59 +08:00
你就不能在 fs 里面加个 go 么
|
2
keepeye 2022-06-10 11:32:07 +08:00 1
跟你一样的环境,一样的代码,我测试可以并发下载
|
3
daokedao OP @shakukansp 抱歉我没看懂,如何在 fs 里面加个 go
|
5
shakukansp 2022-06-10 11:44:25 +08:00
@daokedao 对不起我错了,http 本来就是协程的
|
6
jakes 2022-06-10 11:45:18 +08:00
http.ListenAndServe 里面就有 go c.serve(connCtx),不需要 fs 再 go 了吧?
|
7
beordle 2022-06-10 12:52:45 +08:00 via iPhone
可能说监听队列长度问题。你可以 google 下如何指定。大概率是你 nil 的那个位置。
|
8
pathletboy 2022-06-10 13:09:05 +08:00
我猜是写入了同个文件 NUL ?
|
9
daokedao OP @keepeye @beordle @pathletboy 问题很奇怪,好像和下载的文件大小有关,测试如下:
如果我用小文件(小于 100KB )可以多个同时下载 如果文件大于 1MB 就只能两个同时下载,其他的等待 莫名其妙啊 |
12
dreasky 2022-06-10 15:54:48 +08:00
不会是 cmd 窗口卡住了吧 :doge
|
13
pathletboy 2022-06-10 15:59:01 +08:00
|
14
daokedao OP @dreasky 用同样的 cmd 窗口,从 nginx 下载就没有问题。
@pathletboy 试了 http.DefaultTransport ,也不行。另外我查了 http.DefaultTransport 是用在 client 端的,服务端是用 SetKeepAlivesEnabled(false) ,试了还是不行。 |
15
ToBeHacker 2022-06-10 20:12:55 +08:00 1
nul 文件锁
|
16
Frankcox 2022-06-11 08:03:37 +08:00 1
浏览器这没复现成功,Chrome windows11 go1.18.1 四个标签页下载,都能正常下载。
|
17
AnroZ 2022-06-11 10:13:16 +08:00
go 默认运行在一个 cpu 核上的,尝试设置下 runtime.GOMAXPROCS
|
18
daokedao OP @ToBeHacker 不是 NUL 的问题,即使把 NUL 改为不同的文件名,结果同样。
@AnroZ 网上可查得 Starting from Go 1.5, the default value of GOMAXPROCS is the number of cores. |
19
daokedao OP @Frankcox 我也用 Chrome 试了,连续下载四次,开始时只有前两个下载有进度,后两个下载进度一直显示 0 。
但是过了一会,神奇的事情发生了,第三个下载在一瞬间完成了。又过了一会,第四个也在一瞬间完成了。 而这时,第一个和第二个还在下载中。。。 |
20
daokedao OP @Frankcox 上面一瞬间完成的下载,找到原因了,是下载出错了,只下载了 1KB ,而原文件是 72MB
我又用 Chrome 反复试了,结论还是只能有两个连接同时下载,后面的连接要等前面的结束后才开始下载。 |
21
Kisesy 2022-06-11 18:20:14 +08:00 1
试了一下还真是,第一个和第二个正常下载,第三个就停住不动,如果结束掉其中一个,第三个就正常下载了
|
23
Frankcox 2022-06-11 18:23:57 +08:00 1
@daokedao 我刚刚又试了下,还是没有出现你的问题,同时下载四个 10G 左右的视频。两个视频速度 200mb/s 多,两个视频 50mb/s 左右。其中只有一个视频初始下载速度大概 100B/s 持续了 3 秒左右就正常了。
|
25
Kisesy 2022-06-11 19:02:17 +08:00 1
用 aria2 测试也是这样, 就算是 aria2 和 curl 混用也不能超过 2 个任务, 不知道为什么
|
27
AnroZ 2022-06-12 01:54:55 +08:00 1
刚刚测试了下,的确如此,第三个速度刚开始有速度但马上变为 0 了。
这应该 go http 包实现的问题,有空的话,可以仔细看下实现。 因为我换成 echo 框架,实现也是调用 http.ServeFile 的,测试到 5 个,都没问题。 |
28
haoliang 2022-06-12 11:41:07 +08:00
我觉得能确定的是:即使是单核 go 也能并发处理请求,GOMAXPROCS 并没有带来实质变化。
|
29
haoliang 2022-06-12 11:54:04 +08:00
读了一圈源码,并没有发现问题,敢问楼主啥硬件?
``` for { rw = l.accept(); go Conn(rw).serve(); } ``` 不过 fileServer 没有用 sendfile 而是用 io.copy 让我比较惊讶 |
30
daokedao OP @haoliang 我这台电脑 cpu 是 5700g ,系统是 Windows 11 ,Go 版本是 1.18.1 windows/amd64
|
31
Kisesy 2022-06-12 12:58:42 +08:00
|
32
AnroZ 2022-06-12 23:54:12 +08:00
而且,同样的代码在 WSL 子系统上运行是不存在这个问题
|
33
keepeye 2022-06-13 10:02:39 +08:00
我还是无法复现,curl 也改成 win 下的了,windows terminal + pwsh7 + curl ,4 个窗口同时下载,速度都正常
$ curl --limit-rate 1k http://127.0.0.1:8989/DesktopOK_Installer_x64.zip --output NUL % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 3 594k 3 18944 0 0 1021 0 0:09:56 0:00:18 0:09:38 1025 $ curl --limit-rate 1k http://127.0.0.1:8989/DesktopOK_Installer_x64.zip --output NUL % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 3 594k 3 18944 0 0 1021 0 0:09:56 0:00:18 0:09:38 1023 $ curl --limit-rate 1k http://127.0.0.1:8989/DesktopOK_Installer_x64.zip --output NUL % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 2 594k 2 15872 0 0 1020 0 0:09:57 0:00:15 0:09:42 1020 $ curl --limit-rate 1k http://127.0.0.1:8989/DesktopOK_Installer_x64.zip --output NUL % Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 2 594k 2 15872 0 0 1020 0 0:09:57 0:00:15 0:09:42 1020 |
35
whoami9894 2022-06-13 16:12:44 +08:00 7
http.FileServer 在 Windows 的最底层调用 TransmitFile ( https://github.com/golang/go/blob/master/src/internal/poll/sendfile_windows.go#L61)
TransmitFile 在个人机限制并发数为 2 ,服务器无限制 ( https://docs.microsoft.com/en-us/windows/win32/api/mswsock/nf-mswsock-transmitfile#remarks) > Workstation and client versions of Windows optimize the TransmitFile function for minimum memory and resource utilization by limiting the number of concurrent TransmitFile operations allowed on the system to a maximum of two. On Windows Vista, Windows XP, Windows 2000 Professional, and Windows NT Workstation 3.51 and later only two outstanding TransmitFile requests are handled simultaneously; the third request will wait until one of the previous requests is completed. Server versions of Windows optimize the TransmitFile function for high performance. On server versions, there are no default limits placed on the number of concurrent TransmitFile operations allowed on the system. Expect better performance results when using TransmitFile on server versions of Windows. On server versions of Windows, it is possible to set a limit on the maximum number of concurrent TransmitFile operations by creating a registry entry and setting a value for the following REG_DWORD: HKEY_LOCAL_MACHINE\CurrentControlSet\Services\AFD\Parameters\MaxActiveTransmitFileCount |
36
whoami9894 2022-06-13 16:49:59 +08:00 2
至于为什么 gin 可以,因为它根本没做这个优化
16 年有人发过 pr ( https://github.com/gin-gonic/gin/pull/638),但不知道什么原因关闭了,一直到今天都没实现这个优化 ```go // gin 8 0x000000000054620e in net.(*TCPConn).Write at <autogenerated>:1 9 0x00000000005de533 in net/http.checkConnErrorWriter.Write at d:/go/src/net/http/server.go:3532 10 0x0000000000592e75 in bufio.(*Writer).Write at d:/go/src/bufio/bufio.go:639 11 0x00000000005d2695 in net/http.(*chunkWriter).Write at d:/go/src/net/http/server.go:383 12 0x0000000000592e75 in bufio.(*Writer).Write at d:/go/src/bufio/bufio.go:639 13 0x00000000005d819e in net/http.(*response).write at d:/go/src/net/http/server.go:1592 14 0x00000000005d7ed0 in net/http.(*response).Write at d:/go/src/net/http/server.go:1550 15 0x00000000006f3078 in github.com/gin-gonic/gin.(*responseWriter).Write at c:/users/eddisonwang/go/pkg/mod/github.com/gin-gonic/[email protected]/response_writer.go:78 16 0x0000000000452444 in io.copyBuffer at d:/go/src/io/io.go:425 17 0x00000000004520fa in io.Copy at d:/go/src/io/io.go:382 18 0x00000000004520fa in io.CopyN at d:/go/src/io/io.go:358 19 0x00000000005b2ba5 in net/http.serveContent at d:/go/src/net/http/fs.go:337 20 0x00000000005b4af5 in net/http.serveFile at d:/go/src/net/http/fs.go:664 // std http 6 0x0000000000f0ed5c in net.sendFile at d:/go/src/net/sendfile_windows.go:37 7 0x0000000000f1196e in net.(*TCPConn).readFrom at d:/go/src/net/tcpsock_posix.go:52 8 0x0000000000f10f76 in net.(*TCPConn).ReadFrom at d:/go/src/net/tcpsock.go:104 9 0x0000000000fa08d9 in net/http.(*response).ReadFrom at d:/go/src/net/http/server.go:597 10 0x0000000000e65a8b in io.copyBuffer at d:/go/src/io/io.go:409 11 0x0000000000e657fa in io.Copy at d:/go/src/io/io.go:382 12 0x0000000000e657fa in io.CopyN at d:/go/src/io/io.go:358 13 0x0000000000f80465 in net/http.serveContent at d:/go/src/net/http/fs.go:337 14 0x0000000000f823b5 in net/http.serveFile at d:/go/src/net/http/fs.go:664 ``` |
37
daokedao OP @whoami9894 应该就是这个原因了,佩服 👍👍👍
|
38
Kisesy 2022-06-13 18:15:27 +08:00
@whoami9894 感谢啊,感觉就是有把大锁锁住了文件,原来是这个函数,微软也太抠了,限制到 2
|