带有相对路径字符串的Path.Combine绝对


94

我正在尝试使用结合相对路径的Windows路径Path.Combine

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

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



5
我们在这里得到不同的答案..我不认为这是重复的
CVertex

1
它是重复的,尽管我认为Path.GetFullName是更好的解决方案。
格雷格·迪恩

你只是矛盾自己。但是,谢谢您的替代答案。
CVertex

Answers:


63

工作原理:

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

7
只需确保baseDirectory带有尾随\\,否则您最终将C:\\blah..\\bling.txt无法正常工作。在那种情况下,您可以手动将其添加到字符串中或执行Path.GetFullPath(Path.Combine(baseDirectory, relativePath))
Nelson Rothermel

5
您的“有效作品”部分的结果应该不是C:\bling.txt吗?
cod3monk3y 2014年

为什么基于URI的方法不起作用?根据此答案,结果是有效的(并且似乎在Windows上也可以识别)。
FH



4

对于Windows通用应用程序Path.GetFullPath()不可用,您可以改用System.Uri此类:

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

3

这将为您提供所需的确切信息(路径不必存在才能正常工作)

DirectoryInfo di = new DirectoryInfo(@"C:\blah\..\bling");
string cleanPath = di.FullName;

1
Path.GetFullPath()和DirectoryInfo.FullName都将在虚拟路径上工作。问题是当文件实际存在时,执行过程需要FileIOPermission-两个API都为true。(请参阅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);

0

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));
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.