在 Windows 中获取临时目录名称的最佳方法是什么?我看到我可以使用GetTempPath
andGetTempFileName
来创建一个临时文件,但是有没有相当于 Linux/BSD 的mkdtemp
功能来创建一个临时目录呢?
9 回答
不,没有等效于 mkdtemp。最好的选择是结合使用GetTempPath和GetRandomFileName。
您将需要与此类似的代码:
public string GetTemporaryDirectory()
{
string tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
Directory.CreateDirectory(tempDirectory);
return tempDirectory;
}
我Path.GetTempFileName()
在磁盘上给了我一个有效的伪随机文件路径,然后删除该文件,并创建一个具有相同文件路径的目录。
根据 Chris 对 Scott Dorman 的回答的评论,这避免了检查文件路径是否在一段时间或循环中可用的需要。
public string GetTemporaryDirectory()
{
string tempFolder = Path.GetTempFileName();
File.Delete(tempFolder);
Directory.CreateDirectory(tempFolder);
return tempFolder;
}
如果您确实需要一个加密安全的随机名称,您可能需要调整 Scott 的答案以使用 while 或 do loop 以继续尝试在磁盘上创建路径。
我喜欢使用 GetTempPath()、一个 GUID 创建函数,如 CoCreateGuid() 和 CreateDirectory()。
GUID 被设计为具有很高的唯一性,并且也极不可能有人手动创建与 GUID 具有相同形式的目录(如果他们这样做,则 CreateDirectory() 将无法指示其存在。)
@克里斯。我也沉迷于临时目录可能已经存在的远程风险。关于随机和密码强的讨论也不完全让我满意。
我的方法建立在一个基本事实之上,即 O/S 不能允许 2 次调用来创建文件都成功。.NET 设计人员选择隐藏目录的 Win32 API 功能有点令人惊讶,这使得这更容易,因为当您第二次尝试创建目录时它会返回错误。这是我使用的:
[DllImport(@"kernel32.dll", EntryPoint = "CreateDirectory", SetLastError = true, CharSet = CharSet.Unicode)]
[return: MarshalAs(UnmanagedType.Bool)]
private static extern bool CreateDirectoryApi
([MarshalAs(UnmanagedType.LPTStr)] string lpPathName, IntPtr lpSecurityAttributes);
/// <summary>
/// Creates the directory if it does not exist.
/// </summary>
/// <param name="directoryPath">The directory path.</param>
/// <returns>Returns false if directory already exists. Exceptions for any other errors</returns>
/// <exception cref="System.ComponentModel.Win32Exception"></exception>
internal static bool CreateDirectoryIfItDoesNotExist([NotNull] string directoryPath)
{
if (directoryPath == null) throw new ArgumentNullException("directoryPath");
// First ensure parent exists, since the WIN Api does not
CreateParentFolder(directoryPath);
if (!CreateDirectoryApi(directoryPath, lpSecurityAttributes: IntPtr.Zero))
{
Win32Exception lastException = new Win32Exception();
const int ERROR_ALREADY_EXISTS = 183;
if (lastException.NativeErrorCode == ERROR_ALREADY_EXISTS) return false;
throw new System.IO.IOException(
"An exception occurred while creating directory'" + directoryPath + "'".NewLine() + lastException);
}
return true;
}
您可以决定非托管 p/invoke 代码的“成本/风险”是否值得。大多数人会说不是,但至少你现在有一个选择。
CreateParentFolder() 留给学生作为练习。我使用 Directory.CreateDirectory()。获取目录的父目录时要小心,因为它在根目录时为空。
我通常使用这个:
/// <summary>
/// Creates the unique temporary directory.
/// </summary>
/// <returns>
/// Directory path.
/// </returns>
public string CreateUniqueTempDirectory()
{
var uniqueTempDir = Path.GetFullPath(Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()));
Directory.CreateDirectory(uniqueTempDir);
return uniqueTempDir;
}
如果您想绝对确定临时路径中不存在此目录名称,则需要检查此唯一目录名称是否存在,如果确实存在,请尝试创建另一个目录名称。
但是这种基于 GUID 的实现就足够了。在这种情况下,我没有任何问题的经验。一些 MS 应用程序也使用基于 GUID 的临时目录。
我以这种方式使用了一些答案并实现GetTmpDirectory
了方法。
public string GetTmpDirectory()
{
string tmpDirectory;
do
{
tmpDirectory = Path.Combine(Path.GetTempPath(), Path.GetFileNameWithoutExtension(Path.GetRandomFileName()));
} while (Directory.Exists(tmpDirectory));
Directory.CreateDirectory(tmpDirectory);
return tmpDirectory;
}
这是解决临时目录名称冲突问题的一种更暴力的方法。这不是一个可靠的方法,但它显着减少了文件夹路径冲突的机会。
可以潜在地将其他进程或程序集相关信息添加到目录名称中,以降低冲突的可能性,尽管可能不希望在临时目录名称上显示此类信息。还可以混合时间相关字段的组合顺序,以使文件夹名称看起来更随机。我个人更喜欢这样保留它,因为在调试过程中我更容易找到它们。
string randomlyGeneratedFolderNamePart = Path.GetFileNameWithoutExtension(Path.GetRandomFileName());
string timeRelatedFolderNamePart = DateTime.Now.Year.ToString()
+ DateTime.Now.Month.ToString()
+ DateTime.Now.Day.ToString()
+ DateTime.Now.Hour.ToString()
+ DateTime.Now.Minute.ToString()
+ DateTime.Now.Second.ToString()
+ DateTime.Now.Millisecond.ToString();
string processRelatedFolderNamePart = System.Diagnostics.Process.GetCurrentProcess().Id.ToString();
string temporaryDirectoryName = Path.Combine( Path.GetTempPath()
, timeRelatedFolderNamePart
+ processRelatedFolderNamePart
+ randomlyGeneratedFolderNamePart);
如上所述, Path.GetTempPath() 是一种方法。如果用户设置了 TEMP 环境变量,您也可以调用Environment.GetEnvironmentVariable("TEMP") 。
如果您打算使用 temp 目录作为在应用程序中持久化数据的一种方式,您可能应该考虑使用IsolatedStorage作为配置/状态/等的存储库...
GetTempPath是正确的做法;我不确定您对这种方法的关注是什么。然后,您可以使用CreateDirectory来制作它。