1 回答
TA贡献1821条经验 获得超4个赞
在运行时,我从命令行发出 rm mytest.log 并观察到该程序在下次调用时不会产生错误
WriteString()
是的,这正是指定的行为。该文件也没有被删除。唯一rm
删除的是文件系统中的特定路径条目。单个文件可以有多个路径,也称为硬链接。
只有当文件系统条目(链接)或文件描述符(在程序中打开的文件)对它的最后一次引用已关闭时,才会删除实际文件。
Unix 文件模型的这种特殊行为长期用于实现“未命名”共享内存,通过创建和打开文件,然后/dev/shm
删除文件系统条目——因为这种特殊的做事方式引入了竞争条件,为了安全敏感应用程序引入了新的系统调用,允许创建匿名内存映射,最近 Linux 甚至获得了在文件系统中创建文件的功能,而无需创建路径条目(open
带有O_TMPFILE
标志)。
在较新版本的 Linux 上,您甚至可以为已使用系统调用删除最后一个条目的文件重新/创建文件系统条目linkat
。
更新
问题是,如果最后一个文件系统条目消失了,你真的想报错吗?毕竟这不是一个糟糕的情况,您可以安全地写入和读取,没有问题,请注意,一旦您关闭文件的最后一个文件描述符,它就会丢失。
完全有可能检测到最后一个文件系统条目是否已被删除,如果是,则中止文件操作——但是请注意,这样的代码可能会引入它自己的问题,例如,如果程序希望创建一个新的文件系统条目,使用 linkat 将所有内容正确写入文件后。
无论如何,您可以做的是fstat
-ing 文件(file.Stat
在 Go 中)并查看文件具有的硬链接数。如果该数字降为零,则所有文件系统条目都将消失。
package main
import (
"fmt"
"log"
"os"
"syscall"
"time"
)
func main() {
fmt.Println("Test Operation")
f, err := os.OpenFile("test.txt", os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0644)
if err != nil {
log.Fatal(err)
}
for {
n, err := f.WriteString("blah\n")
if err != nil {
log.Fatal(err)
}
log.Printf("wrote %d bytes\n", n)
time.Sleep(2 * time.Second)
stat, err := f.Stat()
if err != nil{
log.Fatal(err)
}
if sys := stat.Sys(); sys != nil {
if stat, ok := sys.(*syscall.Stat_t); ok {
nlink := uint64(stat.Nlink)
if 0 == nlink {
log.Printf("All filesystem entries to original file removed, exiting")
break
}
}
}
}
}
- 1 回答
- 0 关注
- 120 浏览
添加回答
举报