Go语言tar归档文件的读写操作
怎么入门Golang编程?需要学习哪些知识点?这是新手们刚接触编程时常见的问题;下面golang学习网就来给大家整理分享一些知识点,希望能够给初学者一些帮助。本篇文章就来介绍《Go语言tar归档文件的读写操作》,涉及到文件处理,有需要的可以收藏一下
在上一节《创建 .zip 归档文件》中我们介绍了 zip 归档文件的创建和读取,那么接下来介绍一下 tar 归档文件的创建及读取。创建可压缩的 tar 包
创建 tar 归档文件与创建 .zip 归档文件非常类似,主要不同点在于我们将所有数据都写入相同的 writer 中,并且在写入文件的数据之前必须写入完整的头部,而非仅仅是一个文件名。我们在该 pack 程序的实现中使用了 createTar() 和 writeFileToTar() 函数。
func createTar(filename string, files []string) error { file, err := os.Create(filename) if err != nil { return err } defer file.Close() var fileWriter io.WriterCloser = file if strings.HasSuffix(filename, ".gz") { fileWriter = gzip.NewWriter(file) defer fileWriter.Close() } writer := tar.NewWriter(fileWriter) defer writer.Close() for name := range files { if err := writeFileToTar(writer, name); err != nil { return err } } return nil}该函数创建了包文件,而且如果扩展名显示该 tar 包需要被压缩则添加一个 gzip 过滤。gzip.NewWriter() 函数返回一个 *gzip.Writer 值,它满足 io.WriteCloser 接口(正如打开的 *os.File 一样)。
一旦文件准备好写入,我们创建一个 *tar.Writer 往其中写入数据。然后迭代所有文件并将每一个写入归档文件。
func writeFileToTar(writer *tar.Writer, filename string) error { file, err := os.Open(filename) if err != nil { return err } defer file.Close() stat, err := file.Stat() if err != nil { return err } header := &tar.Header{ Name: sanitizedName(filename), Mode: int64(stat.Mode()), Uid: os.Getuid(), Gid: os.Getuid(), Size: stat.Size(), ModTime: stat.ModTime(), } if err = writer.WriteHeader(header); err != nil { return err } err = io.Copy(writer, file) return err}函数首先打开需要处理的文件并设置延迟关闭。然后调用 Stat() 方法取得文件的模式、大小以及修改日期/时间。这些信息用于填充 *tar.Header,每个文件都必须创建一个 tar.Header 结构并写入到 tar 归档文件里,(此外,我们设置了头部的用户以及组 ID,这会在类 Unix 系统中用到。)我们必须至少设置头部的文件名(其 Name 字段)以及表示文件大小的 Size 字段,否则这个 .tar 包就是非法的。
当 *tar.Header 结构体创建好后,我们将它写入归档文件,再接着写入文件的内容。
解开 tar 归档文件
解开 tar 归档文件比创建 tar 归档文档稍微简单些。然而,跟解开 .zip 文件一样,如果归档文件中的某些文件名包含路径,必须重建目录结构。func unpackTar(filename string) error { file, err := os.Open(filename) if err != nil { return err } defer file.Close() var fileReader io.ReadCloser = file if strings.HasSuffix(filename, ".gz") { if fileReader, err = gzip.NewReader(file); err != nil { return err } defer fileReader.Close() } reader := tar.NewReader(fileReader) return unpackTarFiles(reader)}该方法首先按照 Go语言的常规方式打开归档文件,并延迟关闭它。如果该文件使用了 gzip 压缩则创建一个 gzip 解压缩过滤器并延迟关闭它。gzip.NewReader() 函数返回一个 *gzip.Reader 值,正如打开一个常规文件(类型为 *os.File)一样,它也满足 io.ReadCloser 接口。
设置好了文件 reader 之后,我们创建一个 *tar.Reader 来从中读取数据,并将接下来的工作交给一个辅助函数。
func unpackTarFiles(reader *tar.Reader) error { for { header, err := reader.Next() if err != nil { if err == io.EOF { return nil // OK } return err } filename := sanitizedName(header.Name) switch header.Typeflag { case tar.TypeDir: if err = os.MkdirAll(filename, 0755); err != nil { return err } case tar.TypeReq: if err = unpackTarFile(filename, heaer.Name, reader); err != nil{ return err } } } return nil}该函数使用一个无限循环来迭代读取归档文档中的每一项,直到遇到 io.EOF(或者直到遇到错误)。tar.Next() 方法返回压缩文档中第一项或者下一项的 *tar.Header 结构体,失败则报告错误。如果错误值为 io.EOF,则意味着读取文件结束,因此返回一个空的错误值。
得到 *tar.Header 后,根据该头部的 Name 字段创建一个净化过的文件名。然后,通过该项的类型标记来进行 switch。对于该简单示例,我们只考虑目录和常规文件,但实际上,归档文件中还可以包含许多其他类型的项(如符号链接)。
如果该项是一个目录,我们按照处理 .zip 文件时所采用的方法创建该目录。而如果该项是一个常规文件,我们将其工作交给另一个辅助函数。
func unpackTarFile(filename, tarFilename string, reader *tar.Reader) error{ writer, err := os.Create(filename) if err != nil { return err } defer writer.Close() if err := io.Copy(writer, reader); err != nil { return err } if filename == tarFilename { fmt.Printin(filename) } else { fmt.Printf("%s [%s]", filename, tarFilename) } return nil}针对归档文件中的每一项,该函数创建了一个新文件,并延迟关闭它。然后,它将归档文件的下一项数据复制到该文件中。同时,正如在 unpackZippedFile() 函数中所做的那样,我们将刚创建文件的文件名打印到终端,如果净化过的文件名与原始文件名不同,则在方括号中给出原始文件名。
至此,我们完成了对压缩和归档文件及常规文件处理的介绍。Go语言使用 io.Reader、io.ReadCloser、io.Writer 和 io.WriteCloser 等接口处理文件的方式让开发者可以使用相同的编码模式来读写文件或者其他流(如网络流或者甚至是字符串),从而大大降低了难度。
到这里,我们也就讲完了《Go语言tar归档文件的读写操作》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于golang的知识点!
Go语言使用buffer读取文件
- 上一篇
- Go语言使用buffer读取文件
- 下一篇
- Go语言zip归档文件的读写操作
-
- 鲤鱼花生
- 很好,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢博主分享博文!
- 2023-03-22 19:04:20
-
- 妩媚的小鸭子
- 太详细了,已收藏,感谢up主的这篇技术贴,我会继续支持!
- 2023-03-09 01:50:28
-
- 专注的羊
- 这篇技术贴出现的刚刚好,太细致了,很好,码住,关注作者了!希望作者能多写Golang相关的文章。
- 2023-03-03 19:10:33
-
- Golang · Go教程 | 1星期前 | goroutine · Context · 超时控制 · Go教程 · 后端开发 · Go Goroutine context 超时控制 WithTimeout Done QueryContext
- Go context 超时控制实战:从接口入口到 goroutine 回收的完整流程
- 166浏览 收藏
-
- Golang · Go教程 | 1星期前 | map · 并发安全 · RWMutex · sync.Map · Go教程 · 并发安全 RWMutex sync.Map Go map并发读写 go test race
- Go map 并发读写崩溃怎么办:从复现报错到 RWMutex 修复的完整流程
- 272浏览 收藏
-
- Golang · Go教程 | 2星期前 | singleflight · 并发控制 · Go教程 · 缓存治理 · 接口优化 · Go 并发请求 缓存击穿 singleflight 缓存回填
- Go singleflight 防缓存击穿实战:相同请求只查一次数据库
- 114浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ljg-skills
- ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
- 2608次使用
-
- MELO音乐
- MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
- 2412次使用
-
- UniScribe
- UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
- 2369次使用
-
- 剧云
- 剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
- 2567次使用
-
- 万象有声
- 万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
- 2543次使用
-
- Go语言自定义二进制文件的读写操作
- 2022-12-29 103浏览
-
- Go语言并发目录遍历
- 2023-01-07 137浏览
-
- Python pathlib 批量整理文件实战:按扩展名归档和冲突重命名
- 2026-06-13 166浏览
-
- Go语言使用切片读写文件
- 2023-02-25 190浏览
-
- Python 读取大文件内存飙升复盘:从 read() 一次读入到分块迭代修复
- 2026-06-27 196浏览

