Go 1.16起推荐用os.ReadDir(非递归、高效)或filepath.WalkDir(递归、安全),替代已弃用的ioutil.ReadDir;前者返回fs.DirEntry轻量接口,后者支持深度优先遍历与错误中断。
在 Go 语言中,ioutil.ReadDir 已被弃用(自 Go 1.16 起),推荐使用 os.ReadDir 或 filepath.WalkDir 来遍历目录。下面介绍几种常用、安全且符合现代 Go 实践的方式。
os.ReadDir 是轻量、高效、只读的目录遍历方式,返回 []fs.DirEntry,不加载完整文件信息(如内容或详细系统属性),适合快速列出文件名和基础类型判断。
entry.IsDir() 区分文件与子目录os.ReadDir + os.Stat 组合更高效(避免多次系统调用)示例:
package main
import (
"fmt"
"os"
)
func main() {
entries, err := os.ReadDir(".")
if err != nil {
panic(err)
}
for _, entry := range entries {
if entry.IsDir() {
fmt.Printf("[DIR] %s\n", entry.Name())
} else {
fmt.Printf("[FILE] %s\n", entry.Name())
}
}
}
当需要访问子目录及所有嵌套文件时,filepath.WalkDir 是标准且推荐的方式。它按深度优先顺序遍历,支持中断(返回非 nil error 可停止)。
fs
.DirEntry,无需额外 Stat
filepath.Walk,性能更好、更安全示例(只打印普通文件路径):
package main
import (
"fmt"
"io/fs"
"path/filepath"
)
func main() {
filepath.WalkDir(".", func(path string, d fs.DirEntry, err error) error {
if err != nil {
return nil // 忽略无法访问的路径(如 permission denied)
}
if !d.IsDir() {
fmt.Println("File:", path)
}
return nil
})
}
ioutil.ReadDir 在 Go 1.16 中已被移入 io/ioutil 包并标记为废弃,其底层调用 os.ReadDir 后又对每个条目执行 os.Stat,造成冗余系统调用,性能较差,且返回 []os.FileInfo 类型已不如 fs.DirEntry 轻量灵活。
ioutil.ReadDir,建议直接替换为 os.ReadDir
FileInfo(如修改时间、大小),可对特定条目显式调用 d.Info()
os.ReadDir 返回的条目默认是无序的(取决于文件系统)。如需排序,可手动排序;如需过滤(如只取 .go 文件),用简单条件判断即可:
entries, _ := os.ReadDir(".")
// 过滤并排序
var goFiles []string
for _, e := range entries {
if !e.IsDir() && filepath.Ext(e.Name()) == ".go" {
goFiles = append(goFiles, e.Name())
}
}
sort.Strings(goFiles) // 需 import "sort"
不复杂但容易忽略:记得检查错误、区分目录/文件、优先用 fs.DirEntry 接口而非预加载全部元数据。