8

我正在使用以下代码打开一个文件,用于读取我之前在用户的 %TEMP% 文件夹中创建的文件:

new FileStream(cacheFileName, FileMode.Open, FileAccess.Read, FileShare.Read | FileShare.Delete);

在某些用户的计算机上,这有时会引发 UnauthorizedAccessException 并显示消息“访问路径 ... 被拒绝”。我无法重现这一点。我最初的猜测是防病毒或索引引擎正在做一些时髦的事情,但我也注意到这段代码正在使用“FileShare.Delete”,我不确定它是否应该在那里。

是否存在使用“FileShare.Delete”导致 UnauthorizedAccessException 的情况?

4

2 回答 2

18

是的,FileShare.Delete 往往会导致此问题。由在后台运行并扫描文件的任何程序使用,文件索引器和病毒扫描程序是常见的例子。

FileShare.Delete 允许另一个进程删除该文件,即使后台进程仍然打开该文件并正在读取该文件。其他进程将忘记文件实际上并没有消失,因为它知道文件实际上已被删除。

当其他进程依赖于实际删除的文件并执行其他操作时,麻烦就开始了。通常通过创建具有相同名称的新文件来触发。值得注意的是一种非常不明智的保存文件的方法,因为当保存失败时,如果没有备份,它会导致数据完全丢失,但这种错误很常见。

这将失败,因为文件的目录条目仍然存在,直到打开文件的最后一个进程关闭句柄才会消失。任何其他尝试再次打开文件的进程都将被错误 5“拒绝访问”拍打。包括删除文件并尝试重新创建文件的进程。

解决方法是始终使用“事务性”保存,在尝试覆盖之前重命名文件。在 .NET 中使用 File.Replace(),在本机 winapi 中使用 ReplaceFile()。也很容易手工完成,工作流程是:

  1. 删除备份文件,失败则停止
  2. 将旧文件重命名为备份文件名,如果失败则停止
  3. 使用原始文件名写入新文件,如果失败则重命名备份
  4. 删除备份文件,忽略失败

第 2 步确保永远不会丢失任何数据,如果出现任何问题,原始文件将保持不变。第 4 步确保 FileShare.Delete 将按预期工作,当其他进程关闭其句柄时,该备份文件最终会消失。

于 2013-11-09T13:01:13.213 回答
3

我发现了一个重现这一点的场景:

    static void Main(string[] args)
    {
        string cacheFileName = @"C:\temp.txt";
        using (var filestream = new FileStream(cacheFileName, FileMode.Open, FileAccess.Read, FileShare.Read | FileShare.Delete, 4096, FileOptions.SequentialScan))
        {
            filestream.Read(new byte[100], 1, 1);
            Console.ReadLine();
            GC.KeepAlive(filestream);
        }
        Console.WriteLine("Done!");
    }
}

创建一个“C:\temp.txt”文件,然后运行这个程序。尝试使用 Explorer/TotalCommander 删除文件,它不会抱怨但也不会删除文件。然后,再次运行程序,它会抛出 UnauthorizedAccessException。关闭两个 .exe 后,文件似乎最终被删除。

删除“FileShare.Delete”解决了这个问题,因为它根本不会让您在使用文件时尝试删除它。

于 2013-11-09T11:19:25.507 回答