为了账号安全,请及时绑定邮箱和手机立即绑定

如何检测已删除的文件?

如何检测已删除的文件?

Go
PIPIONE 2023-05-08 14:57:53
写入不存在的文件不会在 Go 中产生错误。例如,这是一个循环写入文件的示例程序:package mainimport (    "log"    "os"    "time")func main() {    f, err := os.OpenFile("mytest.log", os.O_APPEND|os.O_CREATE|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)    }}当它运行时,我rm mytest.log从命令行发出并观察到程序在下一次调用时没有产生错误WriteString()。(我在 Linux 上测试过,其他操作系统可能会有所不同)有没有办法检测文件是否被删除(除了在每次写入之前对文件进行统计)?并且大概写入的字节只是被操作系统丢弃了?
查看完整描述

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

                }

            }

        }

    }

}


查看完整回答
反对 回复 2023-05-08
  • 1 回答
  • 0 关注
  • 120 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信