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())
	}
}
相关推荐
想用offer打牌1 小时前
高并发下如何保证接口的幂等性
后端·面试·状态机
爱勇宝2 小时前
2026一人公司生存指南:用AI大模型,90天跑出你的第一条现金流
前端·后端·架构
golang学习记2 小时前
Go 并发编程:原子操作(Atomics)完全指南
后端
哈里谢顿2 小时前
`127.0.0.1` 和 `0.0.0.0` 有何区别?通过验证 demo来展示
后端
树獭叔叔3 小时前
08-大模型后训练的指令微调SFT:LoRA让大模型微调成本降低99%
后端·aigc·openai
苏三说技术3 小时前
我终于遇到一台真正懂程序员的显示器!
后端
Re_zero3 小时前
线上日志被清空?这段仅10行的 IO 代码里竟然藏着3个毒瘤
java·后端
花落人散处3 小时前
流式输出——解决 HITL 难题 (SpringAIAlibaba)
后端
BingoGo5 小时前
OpenSwoole 26.2.0 发布:支持 PHP 8.5、io_uring 后端及协程调试改进
后端·php
JaguarJack5 小时前
OpenSwoole 26.2.0 发布:支持 PHP 8.5、io_uring 后端及协程调试改进
后端·php·服务端