优化文件下载器的开发方法
来源:stackoverflow
时间:2024-02-25 23:27:25 459浏览 收藏
从现在开始,努力学习吧!本文《优化文件下载器的开发方法》主要讲解了等等相关知识点,我会在golang学习网中持续更新相关的系列文章,欢迎大家关注并积极留言建议。下面就先一起来看一下本篇正文内容吧,希望能帮到你!
我正在尝试提高用 go 实现的下载器的性能。 我认为我遇到了内存使用问题,因为当我尝试时程序卡住了 下载大文件,例如 1gb 或更大。我用它来下载周围的文件 100mgb 和 300mgb 一切都很好。下载器用于显示标头的服务器 接受范围。下面我将向您展示主要的实现和部分内容,但首先让我向您解释一下。
接受范围:字节
在此实现中,我创建了一个 http.client 来设置文件部分的标头范围 我要求的,之后我提出了要求。为了存储我创建的此请求的响应 一个临时文件,并将响应直接复制到该文件中。这样做的想法是避免复制 记忆中的整个反应。这是实现:
func downloadpart(wg *sync.waitgroup, tempname string, url string, part string) { //setting up the client to make the request client := http.client{} request, err := http.newrequest("get", url, nil) //setting up the requests request.header.set("range", part) response, err := client.do(request) checkerror(err, "fatal") defer response.body.close() //creating the temporary file and copying // the response to it file, err := os.create(tempname) checkerror(err, "panic") defer file.close() _, err = io.copy(file, response.body) checkerror(err, "fatal") defer wg.done() }
这个函数在各种 goroutine 中被调用,所以我使用了 waitgroup 来减少计数器 gorputine 结束下载文件的一部分。在所有这些 goroutines 结束之后,我加入了不同的人 单个文件中的临时文件。这就是join函数的实现
func joinfiles(name string) { finalfile, err := os.openfile(name, os.o_create|os.o_append|os.o_wronly, 0644) if err != nil { log.panicln(err.error()) } defer finalfile.close() files, err := ioutil.readdir(".") for _, f := range files { tempdata, err := ioutil.readfile(f.name()) if err != nil { log.panicln(err.error()) } if f.name() != finalfile.name() { finalfile.write(tempdata) os.remove(f.name()) } } }
现在我将向您展示使用这些函数的主函数部分
//start, end and rest are used to set the Range header in the requests //threads are the number of goroutines to used in the download var wg sync.WaitGroup wg.Add(threads) //initializing the goroutines for i := 0; i < threads; i++ { part := fmt.Sprintf("bytes=%d-%d", start, end) start = end + 1 if i == threads-1 { end = end + step + rest } else { end = end + step } go tools.DownloadPart(&wg, fmt.Sprintf("%d.temp", i), url, part) } wg.Wait() log.Println("Joining files...") joinFiles(name)
是否有办法改进此实现?
解决方案
我认为这里最大的问题是如何将文件拼接在一起。调用 ioutil.ReadAll
会将整个文件的内容读入内存,并且由于您对所有部分都执行此操作,因此您可能会在内存中得到整个文件的内容(GC 可能会在中间运行并释放其中的一些内容。)更好的做法是在文件上使用 io.Copy
(在使用 os.Open
打开它之后)将其复制到最终文件中。这样您就不必将内容存储在内存中。
理论要掌握,实操不能落!以上关于《优化文件下载器的开发方法》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习