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())
	}
}
相关推荐
阿丰资源1 分钟前
基于Spring Boot的电影城管理系统(直接运行)
java·spring boot·后端
IT_陈寒3 分钟前
SpringBoot自动配置的坑差点让我加班到天亮
前端·人工智能·后端
xingpanvip4 分钟前
星盘接口开发文档:星相日历接口指南
android·开发语言·前端·css·php·lua
guygg8811 分钟前
基于遗传算法的双层规划模型求解MATLAB实现
开发语言·matlab
凯瑟琳.奥古斯特30 分钟前
SQLAlchemy核心功能解析
开发语言·python·flask
卷Java41 分钟前
GPTQ vs AWQ vs GGUF:模型量化工具横向测评
开发语言·windows·python
charlie1145141911 小时前
嵌入式C++工程实践第20篇:GPIO 输入模式内部电路 —— 芯片是如何“听“到外部信号的
开发语言·c++·stm32·单片机
消失的旧时光-19431 小时前
Spring Boot 工程化进阶:统一返回 + 全局异常 + AOP 通用工具包
java·spring boot·后端·aop·自定义注解
xinhuanjieyi1 小时前
极语言让ai学习的方法
开发语言·学习
xiaogutou11212 小时前
2026年历史课件PPT模板选购指南:教师备课效率与精度的平衡方案
开发语言·c#