将绝对路径和相对路径字符串合并

124

我试图使用Path.Combine方法将Windows路径和相对路径联接。

然而,Path.Combine(@"C:\blah",@"..\bling")返回的是C:\blah\..\bling,而不是C:\bling\

有没有人知道如何在不编写自己的相对路径解析器(这应该不难)的情况下完成此操作?


7
我们这里得到了不同的答案...我认为这不是一个重复的问题。 - CVertex
1
虽然我认为 Path.GetFullName 是更好的解决方案,但它是重复的。 - Greg Dean
你刚才自相矛盾了。 但感谢提供替代答案。 - CVertex
可能是Path.Combine和点符号的重复问题 - Julien Bérubé
8个回答

88

有效的方法:

string relativePath = "..\\bling.txt";
string baseDirectory = "C:\\blah\\";
string absolutePath = Path.GetFullPath(baseDirectory + relativePath);

(结果:absolutePath="C:\bling.txt")

为什么不行

string relativePath = "..\\bling.txt";
Uri baseAbsoluteUri = new Uri("C:\\blah\\");
string absolutePath = new Uri(baseAbsoluteUri, relativePath).AbsolutePath;

(结果:absolutePath="C:/blah/bling.txt")


8
是的,这就是我在帖子中暗示的意思。 - Llyle
9
确保baseDirectory末尾有\,否则你会得到C:\\blah..\\bling.txt这样的结果,而这是无法工作的。在这种情况下,你可以手动将它们添加到字符串中,或者使用Path.GetFullPath(Path.Combine(baseDirectory, relativePath)) - Nelson Rothermel
5
你的“What Works”部分的结果不应该是C:\bling.txt吗? - cod3monk3y
为什么基于URI的方法不起作用?根据这个答案,结果是有效的(在Windows上似乎也被识别了,如此处所示)。 - F-H

51

2
请注意,这仅适用于第一个路径为绝对路径的情况。它不适用于 Path.GetFullPath(Path.Combine(@"..\..\blah",@"\bling")) - derekantrican

19

Path.GetFullPath(@"c:\windows\temp\..\system32")?


4
注意:应该得到 c:\windows\system32 的结果。 - cod3monk3y

5

对于 Windows 通用应用程序,Path.GetFullPath() 不可用,可以使用 System.Uri 类代替:

 Uri uri = new Uri(Path.Combine(@"C:\blah\",@"..\bling"));
 Console.WriteLine(uri.LocalPath);

5

Path.GetFullPath() 无法处理相对路径。

以下解决方案适用于相对和绝对路径,在 Linux 和 Windows 上均可工作,并且保留文本开头的 ..(它们将被规范化)。该解决方案仍依赖于 Path.GetFullPath 并通过小小的变通来进行修复。

这是一个扩展方法,使用方式为 text.Canonicalize()

/// <summary>
///     Fixes "../.." etc
/// </summary>
public static string Canonicalize(this string path)
{
    if (path.IsAbsolutePath())
        return Path.GetFullPath(path);
    var fakeRoot = Environment.CurrentDirectory; // Gives us a cross platform full path
    var combined = Path.Combine(fakeRoot, path);
    combined = Path.GetFullPath(combined);
    return combined.RelativeTo(fakeRoot);
}
private static bool IsAbsolutePath(this string path)
{
    if (path == null) throw new ArgumentNullException(nameof(path));
    return
        Path.IsPathRooted(path)
        && !Path.GetPathRoot(path).Equals(Path.DirectorySeparatorChar.ToString(), StringComparison.Ordinal)
        && !Path.GetPathRoot(path).Equals(Path.AltDirectorySeparatorChar.ToString(), StringComparison.Ordinal);
}
private static string RelativeTo(this string filespec, string folder)
{
    var pathUri = new Uri(filespec);
    // Folders must end in a slash
    if (!folder.EndsWith(Path.DirectorySeparatorChar.ToString())) folder += Path.DirectorySeparatorChar;
    var folderUri = new Uri(folder);
    return Uri.UnescapeDataString(folderUri.MakeRelativeUri(pathUri).ToString()
        .Replace('/', Path.DirectorySeparatorChar));
}

4
这将为您提供所需的内容(路径无需存在即可正常工作)。
DirectoryInfo di = new DirectoryInfo(@"C:\blah\..\bling");
string cleanPath = di.FullName;

1
无论是 Path.GetFullPath() 还是 DirectoryInfo.FullName 都可以用于虚构路径。问题在于当文件实际存在时,执行进程需要 FileIOPermission - 对于这两个 API 都是如此。 (请参阅 MSDN) - Paul Williams

1

小心反斜杠,不要忘记它们(也不要使用两次:)

string relativePath = "..\\bling.txt";
string baseDirectory = "C:\\blah\\";
//OR:
//string relativePath = "\\..\\bling.txt";
//string baseDirectory = "C:\\blah";
//THEN
string absolutePath = Path.GetFullPath(baseDirectory + relativePath);

所有答案和原问题都没有问题,因为它们使用“@”,我们不必双写反斜杠。 - Elo

0

为了处理任何绝对、相对或URI基本路径,似乎没有一种通用的解决方案适用于所有情况...因此,我编写了一个:

    public static String CombinePaths(String basepath, String relpath)
    {
        Stack<String> vs = new Stack<String>();

        int i;

        var s = basepath.Split('\\');
        for (i = 0; i < s.Length; i++)
        {
            if (s[i] != "..")
            {
                if (s[i] != ".")
                    vs.Push(s[i]);
            }
            else
            {
                vs.Pop();
            }
        }

        var r = relpath.Split('\\');
        for (i = 0; i < r.Length; i++)
        {
            if (r[i] != "..")
            {
                if (r[i] != ".")
                    vs.Push(r[i]);
            }
            else
            {
                vs.Pop();
            }
        }

        String ret = "";
        var a = vs.ToArray();
        i = a.Count() - 1;
        while (i > 0)
        {
            ret += a[i].ToString();
            ret += "\\";
            i--;
        }
        ret += a[0].ToString();

        return ret;
    }

随意提高性能或应用任何适用于我的答案的快捷方式(我通常不使用C#,我通常编写C++)。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接