题
如果我的应用程序关闭或崩溃,有什么方法可以确保删除临时文件?理想情况下,我想获取一个临时文件,使用它然后忘记它。
现在我保留一个我的临时文件列表,并使用在Application.ApplicationExit上触发的事件处理程序删除它们。
有更好的方法吗?
解决方案
如果过程被过早杀死,则无法保证,但是,我使用“ using
”这样做..
using System;
using System.IO;
sealed class TempFile : IDisposable
{
string path;
public TempFile() : this(System.IO.Path.GetTempFileName()) { }
public TempFile(string path)
{
if (string.IsNullOrEmpty(path)) throw new ArgumentNullException("path");
this.path = path;
}
public string Path
{
get
{
if (path == null) throw new ObjectDisposedException(GetType().Name);
return path;
}
}
~TempFile() { Dispose(false); }
public void Dispose() { Dispose(true); }
private void Dispose(bool disposing)
{
if (disposing)
{
GC.SuppressFinalize(this);
}
if (path != null)
{
try { File.Delete(path); }
catch { } // best effort
path = null;
}
}
}
static class Program
{
static void Main()
{
string path;
using (var tmp = new TempFile())
{
path = tmp.Path;
Console.WriteLine(File.Exists(path));
}
Console.WriteLine(File.Exists(path));
}
}
现在,当处理 TempFile
或垃圾收集时,文件将被删除(如果可能)。显然,您可以根据需要使用它,或者在某个地方的集合中使用它。
其他提示
考虑使用FileOptions.DeleteOnClose标志:
using (FileStream fs = new FileStream(Path.GetTempFileName(),
FileMode.OpenOrCreate, FileAccess.ReadWrite, FileShare.None,
4096, FileOptions.RandomAccess | FileOptions.DeleteOnClose))
{
// temp file exists
}
// temp file is gone
您可以 P / Invoke CreateFile
并传递 FILE_FLAG_DELETE_ON_CLOSE
标志。这告诉Windows在关闭所有句柄后删除文件。另请参阅: Win32 CreateFile
docs
我会使用.NET TempFileCollection
类,因为它是内置的,可以在旧版本的.NET中使用,并实现 IDisposable
接口,因此可以在之后进行清理如果使用,例如与" using"
关键字一起使用。
这是一个从嵌入式资源中提取文本的示例(通过项目属性页面添加 - >资源选项卡,如下所述:如何在.NET程序集中嵌入文本文件?,然后设置为" EmbeddedResource"
嵌入文件的属性设置)。
// Extracts the contents of the embedded file, writes them to a temp file, executes it, and cleans up automatically on exit.
private void ExtractAndRunMyScript()
{
string vbsFilePath;
// By default, TempFileCollection cleans up after itself.
using (var tempFiles = new System.CodeDom.Compiler.TempFileCollection())
{
vbsFilePath= tempFiles.AddExtension("vbs");
// Using IntelliSense will display the name, but it's the file name
// minus its extension.
System.IO.File.WriteAllText(vbsFilePath, global::Instrumentation.Properties.Resources.MyEmbeddedFileNameWithoutExtension);
RunMyScript(vbsFilePath);
}
System.Diagnostics.Debug.Assert(!File.Exists(vbsFilePath), @"Temp file """ + vbsFilePath+ @""" has not been deleted.");
}
我不是C#程序员,但在C ++中我使用 RAII 为了这。有一些提示在线使用类似RAII的C#行为,但大多数似乎都使用终结器—这不是确定性的。
我认为有一些Windows SDK函数可以创建临时文件,但不知道它们是否在程序终止时自动删除。有 GetTempPath 功能,但文件只有在您注销或重新启动时才会被删除,IIRC。
P.S。 C#析构文档说你可以而且应该在那里发布资源,我觉得有点奇怪。如果是这样,你可以简单地删除析构函数中的临时文件,但同样,这可能不是完全确定的。
很高兴看到你想要负责任,但是如果文件不是很大(> 50MB),那么你将与所有人(包括MS)保持一致,将它们留在临时目录中。磁盘空间很丰富。
正如csl发布的那样,GetTempPath是最佳选择。空间不足的用户将能够运行磁盘清理,并且将清理您的文件(以及其他所有人的文件)。
我使用更可靠的解决方案:
using System.IO;
using System.Reflection;
namespace Konard.Helpers
{
public static partial class TemporaryFiles
{
private const string UserFilesListFilenamePrefix = ".used-temporary-files.txt";
static private readonly object UsedFilesListLock = new object();
private static string GetUsedFilesListFilename()
{
return Assembly.GetEntryAssembly().Location + UserFilesListFilenamePrefix;
}
private static void AddToUsedFilesList(string filename)
{
lock (UsedFilesListLock)
{
using (var writer = File.AppendText(GetUsedFilesListFilename()))
writer.WriteLine(filename);
}
}
public static string UseNew()
{
var filename = Path.GetTempFileName();
AddToUsedFilesList(filename);
return filename;
}
public static void DeleteAllPreviouslyUsed()
{
lock (UsedFilesListLock)
{
var usedFilesListFilename = GetUsedFilesListFilename();
if (!File.Exists(usedFilesListFilename))
return;
using (var listFile = File.Open(usedFilesListFilename, FileMode.Open))
{
using (var reader = new StreamReader(listFile))
{
string tempFileToDelete;
while ((tempFileToDelete = reader.ReadLine()) != null)
{
if (File.Exists(tempFileToDelete))
File.Delete(tempFileToDelete);
}
}
}
// Clean up
using (File.Open(usedFilesListFilename, FileMode.Truncate)) { }
}
}
}
}
每次需要临时文件时使用:
var tempFile = TemporaryFiles.UseNew();
确保在应用程序关闭或崩溃后删除所有临时文件
TemporaryFiles.DeleteAllPreviouslyUsed();
在申请开始时。
您可以在启动时启动一个线程,该线程将删除当他们“不应该”时存在的文件。从崩溃中恢复过来。
如果您正在构建Windows窗体应用程序,则可以使用以下代码:
private void Form1_FormClosing(object sender, FormClosingEventArgs e)
{
File.Delete("temp.data");
}