C#のPathCanonicalizeに相当するものは何ですか?
用途: 2 つのファイル パスが同じファイルを参照しているかどうか (ディスク アクセスなし) を推測する必要があります。私の典型的なアプローチは、MakeAbsolute や PathCanonicalize などのいくつかのフィルターを通過させてから、大文字と小文字を区別しない比較を行うことです。
C#のPathCanonicalizeに相当するものは何ですか?
用途: 2 つのファイル パスが同じファイルを参照しているかどうか (ディスク アクセスなし) を推測する必要があります。私の典型的なアプローチは、MakeAbsolute や PathCanonicalize などのいくつかのフィルターを通過させてから、大文字と小文字を区別しない比較を行うことです。
迅速かつ汚い:
過去に、パス文字列からFileInfoオブジェクトを作成し、FullName プロパティを使用しました。これにより、すべての ..\ と .\ が削除されます。
もちろん、相互運用することもできます:
[DllImport("shlwapi", EntryPoint="PathCanonicalize")]
private static extern bool PathCanonicalize(
StringBuilder lpszDst,
string lpszSrc
);
3 つのソリューション:
呼び出しプロセスがファイルシステムに完全にアクセスできることが 100% 確実な場合の最良のシナリオです。 警告:製品ボックスの許可は難しい場合があります
public static string PathCombineAndCanonicalize1(string path1, string path2)
{
string combined = Path.Combine(path1, path2);
combined = Path.GetFullPath(combined);
return combined;
}
しかし、私たちは常に自由であるとは限りません。多くの場合、許可なく文字列演算を行う必要があります。これにはネイティブコールがあります。 警告:ネイティブ コールに頼る
public static string PathCombineAndCanonicalize2(string path1, string path2)
{
string combined = Path.Combine(path1, path2);
StringBuilder sb = new StringBuilder(Math.Max(260, 2 * combined.Length));
PathCanonicalize(sb, combined);
return sb.ToString();
}
[DllImport("shlwapi.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern bool PathCanonicalize([Out] StringBuilder dst, string src);
3 つ目の戦略は、CLR を騙すことです。Path.GetFullPath() は架空のパスで問題なく機能するため、常にパスを指定していることを確認してください。できることは、偽の UNC パスでルートを交換し、GetFullPath() を呼び出してから、実際のものを元に戻すことです。 注意:これには、コード レビューでハード セルが必要になる場合があります。
public static string PathCombineAndCanonicalize3(string path1, string path2)
{
string originalRoot = string.Empty;
if (Path.IsPathRooted(path1))
{
originalRoot = Path.GetPathRoot(path1);
path1 = path1.Substring(originalRoot.Length);
}
string fakeRoot = @"\\thiscantbe\real\";
string combined = Path.Combine(fakeRoot, path1, path2);
combined = Path.GetFullPath(combined);
combined = combined.Substring(fakeRoot.Length);
combined = Path.Combine(originalRoot, combined);
return combined;
}
Path.GetFullPath()
相対パスでは機能しません。相対パスでも機能するソリューションを探していました。
多くの方法を試しましたが、どれもうまくいきませんでした。@Paulの3番目の戦略はLinuxでは機能せず、相対パスにバグがあり、もう1つのフォルダーが導入され、結果として\\
1つ失われます。..
相対パスと絶対パスの両方で機能するソリューションを次に示します。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));
}