我一直在各处搜索,我似乎无法在任何地方找到这样的选项 - 一旦应用程序终止,使用GetTempFileName创建的临时文件似乎就会被遗忘。
我该怎么做?
答案 0 :(得分:19)
创建文件时,请使用FileOptions.DeleteOnClose
using (var fs = new FileStream(Path.GetTempFileName(), FileMode.Open, FileAccess.ReadWrite, FileShare.None, 4096, FileOptions.DeleteOnClose))
{
// do your work here
}
关闭流后,文件将被删除。
如果您需要在整个时间打开文件,可以使用FileShare.ReadWrite | FileShare.Delete
而不是FileShare.None
打开流,然后在需要访问整个程序中的文件时再次打开相同的文件名。然后在退出程序时关闭流。
注意:您不一定需要显式关闭流,程序关闭的操作将关闭流并执行必要的清理以删除文件。如果您使用这种方法,请确保只要程序处于活动状态,就可以引用活动的流,否则GC可能会清理流并在您之前删除该文件。
更新:如果您有断电,您仍然会在磁盘上有一个文件,但是如果您的程序被End Task强行关闭,该文件仍将被删除。删除是在内核级别而不是应用程序级别处理的。一旦文件的最后一个句柄被关闭,Windows本身就会删除该文件。
虽然您可能无法在断电时删除文件,但还有另一个选项可以最小化磁盘上的临时文件大小。这将要求您更深入地进行更深入的P / Invoke CreateFile
。你可以使用一个标志FILE_ATTRIBUTE_TEMPORARY
,它会做什么告诉windows不要将写入刷新到磁盘,如果可能的话只是将它们保存在缓存中。使用此属性,文件可能会被遗忘,但文件大小可能为0kB。
[DllImport("kernel32.dll", SetLastError = true, CharSet=CharSet.Unicode)]
static extern SafeFileHandle CreateFile(string lpFileName, uint dwDesiredAccess,
uint dwShareMode, IntPtr lpSecurityAttributes, uint dwCreationDisposition,
uint dwFlagsAndAttributes, IntPtr hTemplateFile);
public const uint OPEN_EXISTING = 3
public const uint GENERIC_READ = 0x80000000;
public const uint GENERIC_WRITE = 0x40000000;
public const uint FILE_ATTRIBUTE_TEMPORARY = 0x100;
public const uint FILE_FLAG_DELETE_ON_CLOSE = 0x04000000;
public FileStream OpenTempFile()
{
var path = Path.GetTempFileName();
var handle = CreateFile(path, GENERIC_READ | GENERIC_WRITE, 0, IntPtr.Zero
,OPEN_EXISTING, FILE_ATTRIBUTE_TEMPORARY | FILE_FLAG_DELETE_ON_CLOSE
,IntPtr.Zero);
if (handle.IsInvalid)
{
Marshal.ThrowExceptionForHR(Marshal.GetHRForLastWin32Error());
}
return new FileStream(handle, FileAccess.ReadWrite);
}
上面的代码是在浏览器中编写的,所以可能包含一两个错误,但它可以得到一般的想法。
UPDATE2:即使断电也可以删除文件的一件事是使用WinAPI调用MoveFileEx
,它可以选择在下次重启时删除文件MOVEFILE_DELAY_UNTIL_REBOOT
标记并将IntPtr.Zero
传递给lpNewFileName
。