应使用 os.readdir:它返回有序 direntry 列表,可显式控制递归、过滤隐藏项;优先用 entry.info().modtime() 提取时间,避免硬解析路径;跨设备移动需用 io.copy + os.removeall;并发时按日期分组串行操作或单队列处理。

归档逻辑该用 filepath.Walk 还是 os.ReadDir
直接用 filepath.Walk 容易踩坑:它不保证遍历顺序,且对 symlink 处理不透明,归档时可能跳过子目录或重复处理。而 os.ReadDir(Go 1.16+)返回有序 fs.DirEntry 列表,能显式跳过非目录项、控制递归深度,更适合按需归档。
- 只归档叶子目录?用
os.ReadDir+ 逐层判断entry.IsDir()和子项数量 - 需要跳过隐藏目录(如
.git)?os.ReadDir后过滤entry.Name()[0] == '.' - Go 版本低于 1.16?降级用
os.Open+Readdir,但注意它不区分文件/目录,得额外调用entry.Type()
日期提取别硬解析路径名,优先用文件系统时间戳
用户常把“按日期归档”理解为从目录名里抠 2024-03-15,但实际需求往往是“把今天修改过的目录移到 archive/2024/03/15/”。硬解析路径名既脆弱(命名不规范就崩),又脱离真实归档意图。
- 用
entry.Info().ModTime()获取最后修改时间,再格式化为年/月/日字符串 - 避免用
os.Stat多次调用——os.ReadDir返回的DirEntry已含Info(),复用即可 - 注意时区:默认用本地时区,若需 UTC 归档,用
t.In(time.UTC)再格式化
移动目录前必须检查目标路径是否存在且可写
os.Rename 在跨文件系统时会失败并报 invalid cross-device link,但错误信息不提示具体原因,容易卡在静默失败。
- 先用
os.Stat检查目标父目录(如archive/2024/03/15)是否存在;不存在则用os.MkdirAll创建 - 用
os.WriteFile往目标目录写临时文件测试可写性,比直接Rename更早暴露权限问题 - 跨设备移动必须用
io.Copy+os.RemoveAll模拟,别依赖Rename
并发归档多个目录时,os.Rename 不是线程安全的
多个 goroutine 并发调用 os.Rename 到同一目标路径(比如都往 archive/2024/03/15 移),可能因竞态导致部分目录被覆盖或丢失,且无明确错误提示。
立即学习“go语言免费学习笔记(深入)”;
- 按目标日期分组,每个日期单独起 goroutine,组内串行
Rename - 或用
sync.Mutex锁住目标路径字符串(如mu.Lock(); defer mu.Unlock()),锁粒度是日期而非全局 - 更稳妥的做法:所有移动操作走单个 goroutine 队列,主协程只负责发现和入队,避免任何文件系统竞态
真正麻烦的是时间精度和时区混用——同一个目录在不同机器上 ModTime() 可能差几秒,归档到不同日期;还有 NFS 挂载点的时间戳同步问题。这些没法靠代码完全规避,得提前约定好时钟源和挂载参数。










