Go开发指南:io/ioutil包应用和迁移指南

概览

Go语言的标准库中,ioutil在文件和I/O操作中扮演者重要的角色。虽然从Go 1.16版本开始,Go团队认为ioutil是一个定义不清且难以理解的程序集合,所有功能的实现代码迁移至ioos包。

鉴于ioutil的功能已被标记为废弃,推荐使用新的API,本文在介绍ioutil应用时会相应介绍新的API如何使用。

新旧函数对照表

为确保代码平滑过渡,io/ioutil在Go 1.16+中通过包装新包函数实现兼容。通过阅读io/ioutil/ioutil.goio/ioutil/tempfile.go源码,梳理出新旧版本函数对照表:

原函数 新函数 返回值差异 功能说明
ioutil.Discard io.Discard
ioutil.NopCloser io.NopCloser
ioutil.ReadAll io.ReadAll
ioutil.ReadDir os.ReadDir []fs.FileInfo → []os.DirEntry
ioutil.ReadFile os.ReadFile 读取指定文件的全部内容
ioutil.TempDir os.MkdirTemp 读取指定目录下的文件和目录信息
ioutil.TempFile os.CreateTemp
ioutil.WriteFile os.WriteFile 将指定数据写入指定文件

io/ioutil函数应用

ReadFile

在Go中,读取文件通常需要几个步骤:打开文件->读取文件->关闭文件io/ioutilReadFile函数可以简化该过程,避免手动打开和关闭文件,简化了错误处理,适用于小文件

新旧函数示例代码

go 复制代码
package main

import (
	"fmt"
	"io/ioutil"
	"log"
	"os"
)

func main() {
	var fileName = "data.txt"
	data, err := ioutil.ReadFile(fileName)
	if err != nil {
		log.Fatal(err)
	}
	fmt.Println(string(data))

	// go1.16后该功能由os.ReadFile实现,入参和返回值与旧函数保持一致
	data, err = os.ReadFile(fileName)
	if err != nil {
		log.Fatal(err)
	}
	fmt.Println(string(data))
}

WriteFile

io/ioutilWriteFile将指定数据写入指定文件,如果文件不存在则创建文件后写入,如果文件中含有内容,则会覆盖原有的内容。

新旧函数示例代码

go 复制代码
package main

import (
	"io/ioutil"
	"log"
	"os"
)

func main() {
	var fileName = "data.txt"
	data := []byte("Class: No.1")
	err := ioutil.WriteFile(fileName, data, 0644)
	if err != nil {
		log.Fatal(err)
	}

	// go1.16后该功能由os.WriteFile实现,入参和返回值与旧函数保持一致
	data = []byte("Class: No.2")
	err = os.WriteFile(fileName, data, 0644)
	if err != nil {
		log.Fatal(err)
	}
}

ReadDir

io/ioutilReadDir读取指定目录下所有目录和文件信息,并返回fs.FileInfo数组。新函数则是返回os.DirEntry数组

返回值结构体对比

FileInfo DirEntry 说明
Name() Name() 获取文件或者目录名
IsDir() IsDir() 是否为文件夹
Mode() Type() 类行位
Info() 返回FileInfo
Size() 通过Info()获取FileInfo后再调用Size() 获取文件大小
ModTime() 通过Info()获取FileInfo后再调用ModTime() 获取文件修改时间
Sys() 通过Info()获取FileInfo后再调用Sys()

新旧函数示例代码

go 复制代码
package main

import (
	"fmt"
	"io/ioutil"
	"log"
	"os"
)

func main() {
	path := "../test"
	files, err := ioutil.ReadDir(path)
	if err != nil {
		log.Fatal(err)
	}

	for _, file := range files {
		fmt.Println(file.Sys())
	}
	fmt.Println("------")

	// go1.16后该功能由os.ReadDir实现,入参一致,返回值更改
	newfiles, err := os.ReadDir(path)
	if err != nil {
		log.Fatal(err)
	}

	for _, file := range newfiles {
		info, _ := file.Info()
		fmt.Println(info.Sys())
	}
}
相关推荐
Traving Yu2 分钟前
Spring源码与框架原理
java·后端·spring
王家视频教程图书馆6 分钟前
rust 写gui 程序 最流行的是哪个
开发语言·后端·rust
Wadli9 分钟前
Oncall Agent项目
开发语言
艾莉丝努力练剑14 分钟前
【QT】Qt常用控件与布局管理深度解析:从原理到实践的架构思考
linux·运维·服务器·开发语言·网络·qt·架构
好大哥呀14 分钟前
如何在Spring Boot中配置数据库连接?
数据库·spring boot·后端
杜子不疼.15 分钟前
用 Python 实现 RAG:从文档加载到语义检索全流程
开发语言·人工智能·python
chao18984416 分钟前
基于改进二进制粒子群算法的含需求响应机组组合问题MATLAB实现
开发语言·算法·matlab
lcj251116 分钟前
字符函数,字符串函数,内存函数
c语言·开发语言·c++·windows
独特的螺狮粉16 分钟前
古诗词飞花令随机出题小助手:鸿蒙Flutter框架 实现的古诗词游戏应用
开发语言·flutter·游戏·华为·架构·开源·harmonyos
qq_3961534517 分钟前
docker ddns-go 忘记密码
docker·容器·golang