Go 文件下载需流式读写防内存溢出,支持 Range 断点续传(206响应),用临时文件+重命名保安全,可选校验与超时控制。

Go 语言实现文件下载功能,关键在于合理使用 HTTP 客户端、分块读写、支持 Range 请求头,并持久化记录已下载偏移量。大文件和断点续传不是靠“一次性读完再写”,而是靠“边拉边存 + 断点状态管理”。
不把整个响应体加载进内存,而是用 io.Copy 或带缓冲的 io.CopyN 直接写入文件:
os.O_CREATE | os.O_WRONLY | os.O_TRUNC)io.Copy(dst, resp.Body) 流式写入,不缓存全文resp.Body 和文件句柄(建议用 defer)服务端需支持 Accept-Ranges: bytes(绝大多数静态服务器默认支持)。客户端逻辑如下:
os.Stat 获取已写入字节数 done
req.Header.Set("Range", fmt.Sprintf("bytes=%d-", done))
206 Partial Content
os.O_WRONLY | os.O_APPEND 模式打开文件,从末尾追加写入避免写入中途失败导致文件损坏:
立即学习“go语言免费学习笔记(深入)”;
file.zip.part),完成后再 os.Rename 替换原文件Content-MD5 或自定义 headerhttp.Client.Timeout 和 Transport.MaxIdleConnsPerHost 防连接耗尽核心结构体示例:
type Downloader struct {
Client *http.Client
Path string // 本地保存路径
}
func (d *Downloader) Download(url string) error {
fi, err := os.Stat(d.Path)
var done int64 = 0
if err == nil {
done = fi.Size()
}
req, _ := http.NewRequest("GET", url, nil)
if done > 0 {
req.Header.Set("Range", fmt.Sprintf("bytes=%d-", done))
}
resp, err := d.Client.Do(req)
if err != nil {
return err
}
defer resp.Body.Close()
if done == 0 && resp.StatusCode != http.StatusOK {
return fmt.Errorf("expected 200, got %d", resp.StatusCode)
}
if done > 0 && resp.StatusCode != http.StatusPartialContent {
return fmt.Errorf("expected 206, got %d", resp.StatusCode)
}
f, _ := os.OpenFile(d.Path, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0644)
defer f.Close()
_, err = io.Copy(f, resp.Body)
return err
}以上就是如何使用Golang实现文件下载功能_支持大文件和断点续传的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号