20

有没有办法找到特定类型的文件数,而不必遍历 Directory.GetFiles() 或类似方法的所有结果?我正在寻找这样的东西:

int ComponentCount = MagicFindFileCount(@"c:\windows\system32", "*.dll");

我知道我可以创建一个递归函数来调用 Directory.GetFiles ,但是如果我可以在不进行所有迭代的情况下做到这一点,那将会更加简洁。

编辑:如果不递归和迭代自己就不可能做到这一点,那么最好的方法是什么?

4

7 回答 7

37

您应该使用 Directory.GetFiles() 的Directory.GetFiles(path, searchPattern, SearchOption)重载。

Path 指定路径,searchPattern 指定通配符(例如,*、*.format),SearchOption 提供包含子目录的选项。

此搜索的返回数组的 Length 属性将为您的特定搜索模式和选项提供正确的文件计数:

string[] files = directory.GetFiles(@"c:\windows\system32", "*.dll", SearchOption.AllDirectories);

return files.Length;

编辑:或者,您可以使用Directory.EnumerateFiles 方法

return Directory.EnumerateFiles(@"c:\windows\system32", "*.dll", SearchOption.AllDirectories).Count();
于 2008-08-26T08:53:42.380 回答
8

最巧妙的方法是使用 linq:

var fileCount = (from file in Directory.EnumerateFiles(@"H:\iPod_Control\Music", "*.mp3", SearchOption.AllDirectories)
                    select file).Count();
于 2011-09-15T12:40:27.870 回答
7

您可以使用 GetFiles 的此重载:

Directory.GetFiles 方法(字符串、字符串、搜索选项

和 SearchOption 的这个成员:

AllDirectories - 在搜索操作中包括当前目录和所有子目录。此选项包括重新解析点,例如搜索中的已安装驱动器和符号链接。

GetFiles 返回一个字符串数组,因此您可以获取长度,即找到的文件数。

于 2008-08-26T08:53:50.867 回答
6

我一直在寻找更优化的版本。由于我还没有找到它,我决定将它编码并在这里分享:

    public static int GetFileCount(string path, string searchPattern, SearchOption searchOption)
    {
        var fileCount = 0;
        var fileIter = Directory.EnumerateFiles(path, searchPattern, searchOption);
        foreach (var file in fileIter)
            fileCount++;
        return fileCount;
    }

所有使用 GetFiles/GetDirectories 的解决方案有点慢,因为需要创建所有这些对象。使用枚举,它不会创建任何临时对象 (FileInfo/DirectoryInfo)。

有关详细信息,请参阅备注http://msdn.microsoft.com/en-us/library/dd383571.aspx

于 2011-03-23T18:58:49.610 回答
1

使用递归,您的 MagicFindFileCount 将如下所示:

private int MagicFindFileCount( string strDirectory, string strFilter ) {
     int nFiles = Directory.GetFiles( strDirectory, strFilter ).Length;

     foreach( String dir in Directory.GetDirectories( strDirectory ) ) {
        nFiles += GetNumberOfFiles(dir, strFilter);
     }

     return nFiles;
  }

尽管乔恩的解决方案可能是更好的解决方案。

于 2008-08-26T08:58:03.803 回答
1

我有一个应用程序,它生成父目录中的目录和文件的计数。一些目录包含数千个子目录,每个子目录有数千个文件。为此,在保持响应式用户界面的同时,我执行以下操作(将路径发送到ADirectoryPathWasSelected方法):

public class DirectoryFileCounter
{
    int mDirectoriesToRead = 0;

    // Pass this method the parent directory path
    public void ADirectoryPathWasSelected(string path)
    {
        // create a task to do this in the background for responsive ui
        // state is the path
        Task.Factory.StartNew((state) =>
        {
            try
            {
                // Get the first layer of sub directories
                this.AddCountFilesAndFolders(state.ToString())


             }
             catch // Add Handlers for exceptions
             {}
        }, path));
    }

    // This method is called recursively
    private void AddCountFilesAndFolders(string path)
    {
        try
        {
            // Only doing the top directory to prevent an exception from stopping the entire recursion
            var directories = Directory.EnumerateDirectories(path, "*.*", SearchOption.TopDirectoryOnly);

            // calling class is tracking the count of directories
            this.mDirectoriesToRead += directories.Count();

            // get the child directories
            // this uses an extension method to the IEnumerable<V> interface,
           // which will run a function on an object. In this case 'd' is the 
           // collection of directories
            directories.ActionOnEnumerable(d => AddCountFilesAndFolders(d));
        }
        catch // Add Handlers for exceptions
        {
        }
        try
        {
            // count the files in the directory
            this.mFilesToRead += Directory.EnumerateFiles(path).Count();
        }
        catch// Add Handlers for exceptions
        { }
    }
}
// Extension class
public static class Extensions
{ 
    // this runs the supplied method on each object in the supplied enumerable
    public static void ActionOnEnumerable<V>(this IEnumerable<V> nodes,Action<V> doit)
    {

        foreach (var node in nodes)
        {   
            doit(node);
        }
    }
}
于 2012-04-17T15:16:13.637 回答
0

有人必须做迭代部分。

AFAIK,.NET 中还没有这样的方法,所以我想一定有人是你。

于 2008-08-26T08:44:28.627 回答