StreamWriter.Close()之后的掉电会产生空白文件,为什么?

Siy*_*ion 7 c# streamwriter

好的,这样解释; 我正在开发一个可以在任何时间点出现电源故障的系统,我正在测试的一点是在我使用StreamWriter写出文件之后.代码如下:

// Write the updated file back out to the Shell directory.
using (StreamWriter shellConfigWriter =
        new StreamWriter(@"D:\xxx\Shell\Config\Game.cfg.bak"))
{
    for (int i = 0; i < configContents.Count; i++)
    {
        shellConfigWriter.WriteLine(configContents[i]);
    }
    shellConfigWriter.Close();
}

FileInfo gameCfgBackup = new FileInfo(@"D:\xxx\Shell\Config\Game.cfg.bak");
gameCfgBackup.CopyTo(@"D:\xxx\Shell\Config\Game.cfg", true);
Run Code Online (Sandbox Code Playgroud)

shellConfigWriter(a List字符串)的内容写入用作临时存储的文件,然后将其复制到原始文件上.现在,在此代码完成执行后,电源丢失,再次启动时,文件Game.cfg存在并且大小正确,但是完全空白.起初我认为这是由于在硬盘驱动器上启用了写入缓存,但即使关闭它仍然会发生(尽管不经常).

任何想法都会非常受欢迎!

更新:好的,所以在删除.Close()语句并.Flush()在每次写入操作后调用后,文件仍然是空白的.在创建新文件之前,我可以更进一步创建原始文件的备份,然后我有足够的备份来进行完整性检查,但我认为它不会有助于解决根本问题(当我告诉它写入时,刷新并关闭文件......它没有!).

Jes*_*cer 12

使用对象构造函数的FileOptions参数阻止操作系统缓冲输出FileStream:

    using (Stream fs = new FileStream(@"D:\xxx\Shell\Config\Game.cfg.bak", FileMode.Create, FileAccess.Write, FileShare.None, 0x1000, FileOptions.WriteThrough))
    using (StreamWriter shellConfigWriter = new StreamWriter(fs))
    {
        for (int i = 0; i < configContents.Count; i++)
        {
            shellConfigWriter.WriteLine(configContents[i]);
        }
        shellConfigWriter.Flush();
        shellConfigWriter.BaseStream.Flush();
    }
Run Code Online (Sandbox Code Playgroud)

  • “new FileStream”的替代方法是静态 File.Create 方法:“File.Create(@"D:\xxx\Shell\Config\Game.cfg.bak", 0x1000, FileOptions.WriteThrough, null)”。它的参数较少,在代码中可能看起来更清晰。 (2认同)

Joe*_*orn 7

首先,你不必在那里打电话shellConfigWriter.Close().该using声明将照顾它.您可能想要做的是防止电源故障,而是打电话shellConfigWriter.Flush().


更新
你可能想要考虑的其他事情是,如果电源故障在任何时候都可能真的发生,它可能发生在写入过程中,这样只有一些字节才能进入文件.真的没有办法阻止它.

为了防止这些情况,常见的过程是使用状态/条件标志文件.您使用具有特定名称的零字节文件的文件系统上存在或不存在来告诉程序在恢复时再次拾取的位置.然后,在确定已达到该状态并完成之前的状态之前,不要创建或销毁触发特定状态的文件.

这里的缺点是它可能意味着不时地抛弃大量工作.但好处是它意味着代码的功能部分看起来很正常:要使系统足够健壮,只需要做很多额外的工作.