Path.Combine абсолютний із відносними рядками шляху


94

Я намагаюся приєднати шлях Windows із відносним шляхом, використовуючи Path.Combine.

Однак Path.Combine(@"C:\blah",@"..\bling")повертає C:\blah\..\blingзамість C:\bling\.

Хто-небудь знає, як це зробити, не написавши власний вирішувач відносних шляхів (що не повинно бути надто складно)?



5
Тут ми отримуємо різні відповіді .. Я не думаю, що це дублікат
CVertex

1
це дублікат, хоча я думаю, що Path.GetFullName є кращим рішенням.
Грег Дін

Ви просто суперечили собі. Але дякую за альтернативну відповідь.
CVertex

Відповіді:


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
Так, це те, що я
натякаю

7
Просто переконайтеся, що baseDirectory має кінцевий термін \\, інакше у вас все вийде, C:\\blah..\\bling.txtі це не спрацює. У цьому випадку ви можете вручну додати їх до рядка або зробитиPath.GetFullPath(Path.Combine(baseDirectory, relativePath))
Nelson Rothermel

5
Чи не повинен бути результатом вашого розділу « Що працює»C:\bling.txt ?
cod3monk3y

Чому метод на основі URI не працює? Відповідно до цієї відповіді , результат є дійсним (і, схоже, він розпізнається і в Windows ).
FH

37

Викличте Path.GetFullPath на комбінованому шляху http://msdn.microsoft.com/en-us/library/system.io.path.getfullpath.aspx

> Path.GetFullPath(Path.Combine(@"C:\blah\",@"..\bling"))
C:\bling

(Я згоден, що Path.Combine повинен робити це самостійно)


Зауважте, що це працює, лише якщо перший шлях є абсолютним. Це не працюєPath.GetFullPath(Path.Combine(@"..\..\blah",@"\bling"))
derekantrican


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. (див. MSDN)
Пол Вільямс,

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));
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.