我试图使用Path.Combine
方法将Windows路径和相对路径联接。
然而,Path.Combine(@"C:\blah",@"..\bling")
返回的是C:\blah\..\bling
,而不是C:\bling\
。
有没有人知道如何在不编写自己的相对路径解析器(这应该不难)的情况下完成此操作?
我试图使用Path.Combine
方法将Windows路径和相对路径联接。
然而,Path.Combine(@"C:\blah",@"..\bling")
返回的是C:\blah\..\bling
,而不是C:\bling\
。
有没有人知道如何在不编写自己的相对路径解析器(这应该不难)的情况下完成此操作?
有效的方法:
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")
C:\\blah..\\bling.txt
这样的结果,而这是无法工作的。在这种情况下,你可以手动将它们添加到字符串中,或者使用Path.GetFullPath(Path.Combine(baseDirectory, relativePath))
。 - Nelson RothermelC:\bling.txt
吗? - cod3monk3y在组合的路径上调用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
Path.GetFullPath(@"c:\windows\temp\..\system32")?
c:\windows\system32
的结果。 - cod3monk3y对于 Windows 通用应用程序,Path.GetFullPath()
不可用,可以使用 System.Uri
类代替:
Uri uri = new Uri(Path.Combine(@"C:\blah\",@"..\bling"));
Console.WriteLine(uri.LocalPath);
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));
}
DirectoryInfo di = new DirectoryInfo(@"C:\blah\..\bling");
string cleanPath = di.FullName;
小心反斜杠,不要忘记它们(也不要使用两次:)
string relativePath = "..\\bling.txt";
string baseDirectory = "C:\\blah\\";
//OR:
//string relativePath = "\\..\\bling.txt";
//string baseDirectory = "C:\\blah";
//THEN
string absolutePath = Path.GetFullPath(baseDirectory + relativePath);
为了处理任何绝对、相对或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++)。