os.Create会清空已有内容是因为它默认使用O_CREATE|O_TRUNC|O_WRONLY模式,O_TRUNC导致文件被截断为0字节;避免覆盖应使用O_EXCL,追加写入需用O_appEND。

创建文件时为什么 os.Create 会清空已有内容?
os.Create 的行为是「打开或新建」,但关键点在于它总是以 O_CREATE | O_TRUNC | O_WRONLY 模式调用底层系统调用。这意味着:如果文件已存在,O_TRUNC 会直接截断为 0 字节——不是报错,也不是追加,而是静默清空。
- 想避免覆盖?改用
os.OpenFile(path, os.O_CREATE|os.O_WRONLY|os.O_EXCL, 0644),O_EXCL在文件存在时返回*os.PathError - 想追加写入?用
os.OpenFile(path, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0644) - 创建后立即写入内容,记得调用
file.Close(),否则内容可能未刷盘(尤其在短生命周期程序中)
file, err := os.Create("data.txt") if err != nil { log.Fatal(err) } defer file.Close() file.WriteString("hellon") // 不会自动换行,需手动加 n
删除文件失败常见原因和 os.Remove 的真实行为
os.Remove 只能删除空目录;对非空目录会返回 invalid argument(windows)或 Directory not empty(unix)。它也不会递归删除子项,这点和 shell 的 rm -r 完全不同。
- 删普通文件:直接用
os.Remove("file.txt") - 删非空目录:必须先用
os.RemoveAll("dir/")—— 注意末尾斜杠不影响行为,但路径必须存在 - 删符号链接本身(而非目标):
os.Remove和os.RemoveAll都只删链接,不碰目标 - 权限不足、文件被占用(如 windows 下正在被编辑器打开)、路径含非法字符,都会导致
err != nil,务必检查
err := os.Remove("old.log") if err != nil { if os.IsNotExist(err) { fmt.Println("文件不存在,无需删除") } else { log.Printf("删除失败: %v", err) } }
跨平台路径处理为什么不能硬拼 "dir/filename"?
go 的 os 包内部使用操作系统原生路径分隔符,但硬写 / 或 会导致 Windows 下打开失败(例如 "C:tempfile.txt" 中的 t 被解释为制表符)。
- 永远用
path/filepath.Join("dir", "subdir", "file.txt")拼接路径 - 读取用户输入的路径(如命令行参数),用
filepath.Clean()规范化,它会处理../、重复分隔符、结尾斜杠等 - 判断是否为绝对路径:用
filepath.IsAbs(path),不要用字符串前缀判断(C:在 Windows 是绝对路径,/在 Unix 是)
临时文件和目录该用 ioutil.TempDir 还是 os.MkdirTemp?
ioutil.TempDir 已在 Go 1.16+ 被标记为 deprecated,所有新代码必须用 os.MkdirTemp。两者都生成唯一路径,但后者更安全:
-
os.MkdirTemp("", "prefix-"):第一个参数为""表示使用默认临时目录(os.TempDir()),无需手动指定/tmp或C:Temp - 生成的目录名包含随机后缀,避免竞态条件(两个 goroutine 同时调用不会撞路径)
- 注意:生成的目录是空的,不会自动创建子目录;若需嵌套结构,要自己
os.MkdirAll - 临时文件建议用
os.CreateTemp("", "prefix-*.txt"),同样支持通配符*占位随机字符串
dir, err := os.MkdirTemp("", "test-*.d") if err != nil { log.Fatal(err) } defer os.RemoveAll(dir) // 清理务必放 defer,且放在创建之后
实际项目里最常被忽略的是错误检查粒度——比如用 os.Remove 删除一个预期一定存在的配置文件,却没区分「不存在」和「权限拒绝」,导致故障时难以定位是部署问题还是代码逻辑问题。