filepath.Walk 默认遇错中断,应改用 WalkDir 并对 os.ErrPermission 返回 nil 继续;重命名需用 ReplaceAllString 支持捕获组,提前检查目标路径存在性并处理跨文件系统场景。

用 filepath.Walk 遍历文件时,为什么有些目录进不去?
常见错误是权限拒绝或符号链接循环,filepath.Walk 默认遇到错误会中断整个遍历。实际中更稳妥的做法是用 filepath.WalkDir(Go 1.16+),它允许你控制每个错误是否跳过。
- 遇到
os.ErrPermission直接return nil继续走其他路径 - 对
fs.DirEntry.IsDir()为true的项才做重命名逻辑,避免误操作文件 - 注意:Windows 下长路径或
\?前缀可能绕过默认限制,但filepath.WalkDir不自动处理,需提前用filepath.Clean规范化
正则替换用 regexp.ReplaceAllString 还是 ReplaceAllStringFunc?
二者行为不同:ReplaceAllString 替换匹配到的**整个子串**,而 ReplaceAllStringFunc 接收一个函数,对每个匹配结果做任意变换——这对重命名最关键:你需要保留原文件名中未被正则捕获的部分。
- 比如想把
IMG_20230101_123456.jpg改成2023-01-01_123456.jpg,得靠捕获组(d{4})(d{2})(d{2})+$1-$2-$3,这时必须用Regexp.ReplaceAllString配合String参数里的$n引用 -
ReplaceAllStringFunc没有捕获组支持,只适合简单全量替换(如全部转小写) - 别忘了编译正则时加
^(?i)控制大小写,否则.jpg和.JPG得写两遍规则
重命名前不检查目标文件是否存在,会直接覆盖
Go 的 os.Rename 在目标路径已存在时,Linux/macOS 返回 syscall.EBUSY 或 os.ErrExist(取决于系统),Windows 可能静默失败或覆盖——这和用户直觉严重不符。
- 务必在调用
os.Rename前用os.Stat检查目标路径:if _, err := os.Stat(newPath); err == nil { /* 存在,报错或加序号 */ } - 批量场景下推荐加序号后缀,例如
file(1).txt,而不是终止整个流程 - 注意:
os.Rename跨文件系统会失败,此时要 fallback 到io.Copy+os.Remove,但得自己处理原子性(比如先写临时文件再 rename)
命令行参数解析选 flag 还是第三方库?
对于轻量工具,标准库 flag 完全够用,且无额外依赖。但要注意几个易错点:
立即学习“go语言免费学习笔记(深入)”;
-
flag.String返回的是*string,别忘了解引用:*patternFlag - 路径参数必须用
flag.Arg(0)获取,不能混在flag.String里——否则用户输rename . 's/old/new/'会被当成 flag 解析失败 - 正则标志位(如
-i)建议用flag.Bool单独控制,而不是塞进 pattern 字符串里,否则用户容易写错格式(比如漏掉(?i)开头)
真正麻烦的不是写正则,而是路径拼接时忘记用 filepath.Join;也不是遍历,而是没意识到 os.Rename 在不同系统语义不一致。这些细节不手动试一遍,光看文档根本记不住。










