Given full path, check if path is subdirectory of some other path, or otherwise

前端 未结 9 1344
难免孤独
难免孤独 2020-12-03 13:22

I have 2 strings - dir1 and dir2, and I need to check if one is sub-directory for other. I tried to go with Contains method:

dir1.contains(dir2);
         


        
9条回答
  •  离开以前
    2020-12-03 14:11

    • Case insensitive
    • Tolerates mix of \ and / folder delimiters
    • Tolerates ..\ in path
    • Avoids matching on partial folder names (c:\foobar not a subpath of c:\foo)

    Note: This only matches on the path string and does not work for symbolic links and other kinds of links in the filesystem.

    Code:

    public static class StringExtensions
    {
        /// 
        /// Returns true if  starts with the path .
        /// The comparison is case-insensitive, handles / and \ slashes as folder separators and
        /// only matches if the base dir folder name is matched exactly ("c:\foobar\file.txt" is not a sub path of "c:\foo").
        /// 
        public static bool IsSubPathOf(this string path, string baseDirPath)
        {
            string normalizedPath = Path.GetFullPath(path.Replace('/', '\\')
                .WithEnding("\\"));
    
            string normalizedBaseDirPath = Path.GetFullPath(baseDirPath.Replace('/', '\\')
                .WithEnding("\\"));
    
            return normalizedPath.StartsWith(normalizedBaseDirPath, StringComparison.OrdinalIgnoreCase);
        }
    
        /// 
        /// Returns  with the minimal concatenation of  (starting from end) that
        /// results in satisfying .EndsWith(ending).
        /// 
        /// "hel".WithEnding("llo") returns "hello", which is the result of "hel" + "lo".
        public static string WithEnding([CanBeNull] this string str, string ending)
        {
            if (str == null)
                return ending;
    
            string result = str;
    
            // Right() is 1-indexed, so include these cases
            // * Append no characters
            // * Append up to N characters, where N is ending length
            for (int i = 0; i <= ending.Length; i++)
            {
                string tmp = result + ending.Right(i);
                if (tmp.EndsWith(ending))
                    return tmp;
            }
    
            return result;
        }
    
        /// Gets the rightmost  characters from a string.
        /// The string to retrieve the substring from.
        /// The number of characters to retrieve.
        /// The substring.
        public static string Right([NotNull] this string value, int length)
        {
            if (value == null)
            {
                throw new ArgumentNullException("value");
            }
            if (length < 0)
            {
                throw new ArgumentOutOfRangeException("length", length, "Length is less than zero");
            }
    
            return (length < value.Length) ? value.Substring(value.Length - length) : value;
        }
    }
    

    Test cases (NUnit):

    [TestFixture]
    public class StringExtensionsTest
    {
        [TestCase(@"c:\foo", @"c:", Result = true)]
        [TestCase(@"c:\foo", @"c:\", Result = true)]
        [TestCase(@"c:\foo", @"c:\foo", Result = true)]
        [TestCase(@"c:\foo", @"c:\foo\", Result = true)]
        [TestCase(@"c:\foo\", @"c:\foo", Result = true)]
        [TestCase(@"c:\foo\bar\", @"c:\foo\", Result = true)]
        [TestCase(@"c:\foo\bar", @"c:\foo\", Result = true)]
        [TestCase(@"c:\foo\a.txt", @"c:\foo", Result = true)]
        [TestCase(@"c:\FOO\a.txt", @"c:\foo", Result = true)]
        [TestCase(@"c:/foo/a.txt", @"c:\foo", Result = true)]
        [TestCase(@"c:\foobar", @"c:\foo", Result = false)]
        [TestCase(@"c:\foobar\a.txt", @"c:\foo", Result = false)]
        [TestCase(@"c:\foobar\a.txt", @"c:\foo\", Result = false)]
        [TestCase(@"c:\foo\a.txt", @"c:\foobar", Result = false)]
        [TestCase(@"c:\foo\a.txt", @"c:\foobar\", Result = false)]
        [TestCase(@"c:\foo\..\bar\baz", @"c:\foo", Result = false)]
        [TestCase(@"c:\foo\..\bar\baz", @"c:\bar", Result = true)]
        [TestCase(@"c:\foo\..\bar\baz", @"c:\barr", Result = false)]
        public bool IsSubPathOfTest(string path, string baseDirPath)
        {
            return path.IsSubPathOf(baseDirPath);
        }
    }
    

    Updates

    • 2015-08-18: Fix bug matching on partial folder names. Add test cases.
    • 2015-09-02: Support ..\ in paths, add missing code
    • 2017-09-06: Add note on symbolic links.

提交回复
热议问题