Найти количество файлов с определенным расширением во всех подкаталогах - PullRequest
19 голосов
/ 26 августа 2008

Есть ли способ узнать количество файлов определенного типа без необходимости циклически просматривать все результаты в Directory.GetFiles () или аналогичном методе? Я ищу что-то вроде этого:

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

Я знаю, что могу сделать рекурсивную функцию для вызова Directory.GetFiles, но было бы намного чище, если бы я мог сделать это без всей итерации.

РЕДАКТИРОВАТЬ: Если это невозможно сделать без повторения и итерации, что было бы лучшим способом сделать это?

Ответы [ 7 ]

35 голосов
/ 26 августа 2008

Вы должны использовать Directory.GetFiles (path, searchPattern, SearchOption) перегрузку Directory.GetFiles ().

Путь указывает путь, 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();
8 голосов
/ 15 сентября 2011

Самый удобный способ использовать linq:

var fileCount = (from file in Directory.EnumerateFiles(@"H:\iPod_Control\Music", "*.mp3", SearchOption.AllDirectories)
                    select file).Count();
7 голосов
/ 26 августа 2008

Вы можете использовать эту перегрузку GetFiles:

Метод Directory.GetFiles (String, Строка, SearchOption )

и этот член SearchOption:

AllDirectories - Включает текущий каталог и все подкаталоги в поисковой операции. Этот вариант включает точки повторной обработки, такие как смонтированные диски и символические ссылки в поиск.

GetFiles возвращает массив строк, поэтому вы можете просто получить длину, которая является количеством найденных файлов.

6 голосов
/ 23 марта 2011

Я искал более оптимизированную версию. Поскольку я не нашел его, я решил написать его и поделиться им здесь:

    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 для получения дополнительной информации

1 голос
/ 17 апреля 2012

У меня есть приложение, которое генерирует количество каталогов и файлов в родительском каталоге. Некоторые из каталогов содержат тысячи подкаталогов с тысячами файлов в каждом. Чтобы сделать это, поддерживая отзывчивый пользовательский интерфейс, я делаю следующее (отправляя путь в метод 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);
        }
    }
}
1 голос
/ 26 августа 2008

При использовании рекурсии ваш 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;
  }

Хотя Решение Джона может быть лучше.

0 голосов
/ 26 августа 2008

Кто-то должен выполнить итерационную часть.

AFAIK, такого метода уже нет в .NET, поэтому я предполагаю, что кто-то должен быть вами.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...