Go语言可用goroutine+channel实现并发下载,通过http.Client设超时、限流控制并发数、ProgressWriter实时更新进度、Downloader封装任务管理,并支持断点续传、自动建目录、SHA256校验及termenv/pb/v3美化终端输出。
Go 语言原生支持并发,用 goroutine + channel 即可轻松实现多文件并行下载。核心思路是:为每个文件启动一个 goroutine 执行 HTTP GET 请求,把下载结果(成功/失败、文件名、耗时等)通过 channel 汇总到主 goroutine。
关键点:
http.Client 并设置合理的超时(如 Timeout: 30 * time.Second),避免单个卡死拖垮整体semaphore 模拟信号量resp.Body),记得调用 defer resp.Body.Close()
HTTP 响应头中通常包含 Content-Length,可据此预估总大小;配合读取响应体时的累计字节数,就能算出当前进度百分比。Go 中推荐用 io.Copy 配合自定义的 io.Writer(实现 Write 方法),在每次写入时更新计数器并触发进度回调。
简单做法示例:
ProgressWriter{Total int64, Written int64, OnProgress func(int64, int64)}
Write(p []byte) (n int, err error),累
加 Written += int64(len(p)),再调用 OnProgress(Written, Total)
io.Copy(progressWriter, resp.Body)
终端显示可用 \r 回车不换行刷新同一行,例如:fmt.Printf("\r%s: %d/%d (%.1f%%)", filename, written, total, float64(written)*100/float64(total))
建议封装一个 Downloader 类型,持有并发控制、任务队列、进度汇总等逻辑。每个下载任务用结构体表示,含 URL、保存路径、是否完成、错误信息等字段。
常用增强项:
Content-Length,若匹配则跳过;否则用 Range 请求头继续下载(需服务端支持)os.MkdirAll(filepath.Dir(dest), 0755) 确保路径存在ETag 或额外提供 checksum)纯命令行下,可用 github.com/muesli/termenv 或 github.com/cheggaaa/pb/v3 实现带颜色、动态进度条的输出。后者更轻量,支持多任务并行条形图:
pb.Full.Start(),每任务创建一个 pb.ProgressBar,设置 Total 和 SetUnits(pb.U_BYTES)
progressBar.NewProxyReader(resp.Body) 作为 io.Reader 传给 io.Copy
无需 GUI,也能做出专业级的命令行下载体验。