145

在 Windows 中获取临时目录名称的最佳方法是什么?我看到我可以使用GetTempPathandGetTempFileName来创建一个临时文件,但是有没有相当于 Linux/BSD 的mkdtemp功能来创建一个临时目录呢?

4

9 回答 9

270

不,没有等效于 mkdtemp。最好的选择是结合使用GetTempPathGetRandomFileName

您将需要与此类似的代码:

public string GetTemporaryDirectory()
{
   string tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
   Directory.CreateDirectory(tempDirectory);
   return tempDirectory;
}
于 2008-11-10T16:55:57.770 回答
29

Path.GetTempFileName()在磁盘上给了我一个有效的伪随机文件路径,然后删除该文件,并创建一个具有相同文件路径的目录。

根据 Chris 对 Scott Dorman 的回答的评论,这避免了检查文件路径是否在一段时间或循环中可用的需要。

public string GetTemporaryDirectory()
{
  string tempFolder = Path.GetTempFileName();
  File.Delete(tempFolder);
  Directory.CreateDirectory(tempFolder);

  return tempFolder;
}

如果您确实需要一个加密安全的随机名称,您可能需要调整 Scott 的答案以使用 while 或 do loop 以继续尝试在磁盘上创建路径。

于 2013-12-07T20:01:11.943 回答
8

我喜欢使用 GetTempPath()、一个 GUID 创建函数,如 CoCreateGuid() 和 CreateDirectory()。

GUID 被设计为具有很高的唯一性,并且也极不可能有人手动创建与 GUID 具有相同形式的目录(如果他们这样做,则 CreateDirectory() 将无法指示其存在。)

于 2010-08-06T04:59:18.273 回答
6

@克里斯。我也沉迷于临时目录可能已经存在的远程风险。关于随机和密码强的讨论也不完全让我满意。

我的方法建立在一个基本事实之上,即 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()。获取目录的父目录时要小心,因为它在根目录时为空。

于 2016-01-10T22:42:45.930 回答
6

我通常使用这个:

    /// <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 的临时目录。

于 2016-09-02T07:25:32.983 回答
2

我以这种方式使用了一些答案并实现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;
}
于 2020-12-14T14:24:01.263 回答
1

这是解决临时目录名称冲突问题的一种更暴力的方法。这不是一个可靠的方法,但它显着减少了文件夹路径冲突的机会。

可以潜在地将其他进程或程序集相关信息添加到目录名称中,以降低冲突的可能性,尽管可能不希望在临时目录名称上显示此类信息。还可以混合时间相关字段的组合顺序,以使文件夹名称看起来更随机。我个人更喜欢这样保留它,因为在调试过程中我更容易找到它们。

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);
于 2013-12-09T23:48:49.183 回答
0

如上所述, Path.GetTempPath() 是一种方法。如果用户设置了 TEMP 环境变量,您也可以调用Environment.GetEnvironmentVariable("TEMP") 。

如果您打算使用 temp 目录作为在应用程序中持久化数据的一种方式,您可能应该考虑使用IsolatedStorage作为配置/状态/等的存储库...

于 2008-11-10T17:05:46.127 回答
-1

GetTempPath是正确的做法;我不确定您对这种方法的关注是什么。然后,您可以使用CreateDirectory来制作它。

于 2008-11-10T16:55:18.987 回答