使用Golang实现HTTP文件下载需避免内存溢出,小文件可用http.Get配合io.ReadAll和os.WriteFile;大文件应采用流式下载,通过http.Get获取响应后,用io.Copy将resp.Body分块写入由os.Create创建的文件中,确保内存占用恒定;生产环境推荐自定义http.Client设置超时,并通过包装Reader实现进度反馈,结合Content-Length预估大小,支持认证时需手动构造Request并设置Header,禁止复用DefaultClient以防并发问题。
使用Golang实现HTTP文件下载很简单,核心是用http.Get发起请求,再把响应体流式写入本地文件,避免内存爆满。关键在于处理错误、设置超时、校验状态码和进度反馈。
适用于几MB以内的文件,代码简洁:
http.Get获取响应,检查resp.StatusCode是否为200io.ReadAll读取全部响应体(注意:大文件会吃光内存!)os.WriteFile一次性保存到磁盘示例:
resp, err := http.Get("https://example.com/file.zip")这才是生产环境该用的方式,内存占用恒定(约几十KB),支持GB级文件:
os.Create新建文件句柄io.Copy把resp.Body直接拷贝到文件,底层自动分块读写defer f.Close()和defer resp.Body.Close())示例:
resp, err := http.Get("https://example.com/large.iso")
s.Create("large.iso")真实场景需防卡死、要感知进度:
http.Client自定义Timeout或Context.WithTimeout
resp.Body为自定义Reader,每次Read时更新计数器fmt.Print("\r%3.1f%%", float64(n)/size*100)做简单终端刷新小技巧:用resp.Header.Get("Content-Length")获取预估总大小(非强制字段,可能为空)。
比如需要Bearer Token或Cookie:
http.Client,手动构造http.Request
req.Header.Set("Authorization", "Bearer xxx")或req.AddCookie(...)
client.Do(req)发送注意:不要复用http.DefaultClient在高并发下设Header,应新建Client或用WithCancel Context隔离。
基本上就这些。不复杂但容易忽略错误处理和流控——尤其是大文件千万别用ReadAll。