登录
首页 >  Golang >  Go问答

如何恢复已删除的文件?

来源:stackoverflow

时间:2024-03-08 15:33:36 489浏览 收藏

目前golang学习网上已经有很多关于Golang的文章了,自己在初次阅读这些文章中,也见识到了很多学习思路;那么本文《如何恢复已删除的文件?》,也希望能帮助到大家,如果阅读完后真的对你学习Golang有帮助,欢迎动动手指,评论留言并分享~

问题内容

写入不存在的文件不会在 go 中产生错误。

例如,以下是循环写入文件的示例程序:

package main

import (
    "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上测试过,其他操作系统可能会有所不同)

有没有办法检测文件是否被删除(除了在每次写入之前对文件进行统计之外)?想必写入的字节会被操作系统简单地丢弃?


解决方案


运行时,我从命令行发出 rm mytest.log 并观察到程序在下次调用 writestring() 时不会产生错误

是的,这正是指定的行为。该文件还没有被删除。 rm 唯一删除的是文件系统中的特定路径条目。单个文件可以有多个路径,也称为硬链接

仅当通过文件系统条目(链接)或文件描述符(在程序中打开的文件)最后一次引用该文件时,实际文件才会被删除。

unix 文件模型的这种特殊行为很长一段时间以来都被用来实现“未命名”共享内存,方法是在 /dev/shm 中创建并打开一个文件,然后删除文件系统条目 - 因为这种特殊的做法things 引入了竞争条件,对于安全敏感的应用程序引入了新的系统调用,允许创建匿名内存映射,最近 linux 甚至获得了在文件系统中创建文件的函数,而无需创建路径条目(带有 o_tmpfile 标志的 open)。

在较新版本的 linux 上,您甚至可以为已使用 linkat 系统调用删除最后一个条目的文件重新/创建文件系统条目。

更新

问题是,如果最后一个文件系统条目消失,您真的想出错吗?毕竟这不是一个坏情况,您可以安全地写入和读取,没有问题,只是要注意,一旦关闭文件的最后一个文件描述符,它就会丢失。

完全有可能检测最后一个文件系统条目是否已被删除,如果是,则中止文件操作 - 但请注意,此类代码可能会引入它自己的问题,例如,如果程序希望创建一个新的文件系统文件系统条目,一旦所有内容都正确写入文件,使用 linkat。

无论如何,你能做的就是 fstat-ing 文件(go 中的 file.stat)并查看文件具有的硬链接数量。如果该数字降至零,则所有文件系统条目都会消失。实际上在 go 中获取这个数字有点棘手,这里有描述 Counting hard links to a file in 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
                }
            }
        }
    }
}

以上就是《如何恢复已删除的文件?》的详细内容,更多关于的资料请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>