Удобный для начинающих способ получить список всех файлов и каталогов - PullRequest
5 голосов
/ 10 октября 2009

Используя .NET 3.0, у меня есть метод, приведенный ниже, который корректно возвращает коллекцию всех файлов и каталогов (и подкаталогов) указанного каталога. Я хотел бы, если это возможно, замять это, чтобы использовать только те конструкции, с которыми мне вполне комфортно. В частности, вот что мне не ясно:

1. IEnumerable<FileSystemInfo>: I'd like to return List<FileSystemInfo> instead
2. Stack<FileSystemInfo>: I'd list to use List<FileSystemInfo> instead.
3. yield return: I've never used this before

.

public static IEnumerable<FileSystemInfo> GetAllFilesAndDirectories ( string dir ) {

    DirectoryInfo dirInfo = new DirectoryInfo( dir );
    Stack<FileSystemInfo> stack = new Stack<FileSystemInfo>();

    stack.Push( dirInfo );
    while ( dirInfo != null || stack.Count > 0 ) {
        FileSystemInfo fileSystemInfo = stack.Pop();
        DirectoryInfo subDirectoryInfo = fileSystemInfo as DirectoryInfo;
        if ( subDirectoryInfo != null ) {
            yield return subDirectoryInfo;
            foreach ( FileSystemInfo fsi in subDirectoryInfo.GetFileSystemInfos() ) {
                stack.Push( fsi );
            }
            dirInfo = subDirectoryInfo;
        } else {
            yield return fileSystemInfo;
            dirInfo = null;
        }
    }

}

.

Можно выдвинуть аргумент, что я должен просто освоиться с приведенным выше кодом, но это не то, что я снимаю на сегодня.

Заранее спасибо

Ответы [ 6 ]

2 голосов
/ 10 октября 2009

Вот кратчайший путь, который я могу придумать:

static List<FileSystemInfo> GetAllFilesAndDirectories(string dir)
{
    DirectoryInfo dirInfo = new DirectoryInfo(dir);            
    List<FileSystemInfo> allFilesAndDirectories = new List<FileSystemInfo>();

    allFilesAndDirectories.AddRange(dirInfo.GetFiles("*", SearchOption.AllDirectories));
    allFilesAndDirectories.AddRange(dirInfo.GetDirectories("*", SearchOption.AllDirectories));

    return allFilesAndDirectories;
}

Возвращает список всех файлов и каталогов на всех уровнях, начиная с указанного пути. В порядке, в котором они возвращаются, будут все файлы, а затем все каталоги.

1 голос
/ 10 октября 2009
    public List<Object> GetFilesAndDirectories(string path)
    {
        List<Object> lst = new List<Object>();
        string[] dirs = null;

        try
        {
            dirs = Directory.GetDirectories(path);
        }
        catch (Exception e)
        {
            Console.WriteLine(e.Message);
        }

        foreach (string d in dirs)
        {
            string[] files = null;

            try
            {
                files = Directory.GetFiles(d);
            }
            catch (Exception e)
            {
                Console.WriteLine(e.Message);
            }

            foreach (string f in files)
            {
                lst.Add(f);
            }

            lst.Add(d);

            lst.AddRange(GetFilesAndDirectories(d));
        }

        return lst;
    }


List<Object> stuff = GetFilesAndDirectories(someRoot);
1 голос
/ 10 октября 2009

Я полагаю, вы ищете существующий метод GetFileSystemInfos (строка, SearchOptions). Если вы укажете AllDirectories в качестве значения SearchOptions, он будет рекурсивно искать в переданной папке.

Например:

public static List<FileSystemInfo> GetAllFilesAndDirectories ( string dir ) {
  DirectoryInfo info = new DirectoryInfo(dir);
  FileSystemInfo[] all = info.GetFileSystemInfos("*", SearchOptions.AllDirectories);
  return new List<FileSystemInfo>(all);
}

Если вы хотите записать это далеко, вы можете сделать следующее

public static List<FileSystemInfo> GetAllFilesAndDirectories ( string dir ) {
  int i = 0; 
  List<DirectoryInfo> toProcess = new List<DirectoryInfo>();
  List<FileSystemInfo> list = new List<FileSystemInfo>();
  toProcess.Add(new DirectoryInfo(dir));
  while ( i < toProcess.Count ) { 
    DirectoryInfo curDir = toProcess[i];
    foreach ( FileSystemInfo curFile in curDir.GetFileSystemInfos() ) {
      list.Add(curFile);
      DirectoryInfo maybe = curFile as DirectoryInfo;
      if ( maybe != null ) {
        toProcess.Add(maybe);
      }
    i++;
  }
  return list;
}

  FileSystemInfo[] all = info.GetFileSystemInfos("*", SearchOptions.AllDirectories);
  return new List<FileSystemInfo>(all);
}
0 голосов
/ 10 октября 2009

Я думаю, с точки зрения читабельности кода, вам лучше всего написать рекурсивную функцию. Рекурсивная функция - это та, которая вызывает сама себя, пока не достигнет точки, когда ей не нужно вызывать никакую другую функцию.

Чтобы проиллюстрировать факториал n, записанный как n! и определенная как величина 1 x 2 x 3 x ... x n (где n - положительное целое число), может быть довольно легко определена рекурсивным способом следующим образом.

public int factorial(int n)
{
    if (n < 0)
    {
        throw new Exception("A factorial cannot be calculated for negative integers.");
    }

    if (n == 0 || n == 1)
    {
        // end condition, where we do not need to make a recursive call anymore
        return 1;
    }
    else
    {
        // recursive call
        return n * factorial(n - 1);
    }
}

Примечание: 0! и 1! определены как 1.

Аналогично, метод для перечисления всех файлов и папок по заданному пути также может быть определен рекурсивно. Это потому, что файлы и папки имеют рекурсивную структуру.

Поэтому такой метод будет работать:

public static List<FileSystemInfo> GetAllFilesAndFolders(string folder)
{
    // NOTE : We are performing some basic sanity checking
    // on the method's formal parameters here
    if (string.IsNullOrEmpty(folder))
    {
        throw new ArgumentException("An empty string is not a valid path.", "folder");
    }
    if (!Directory.Exists(folder))
    {
        throw new ArgumentException("The string must be an existing path.", "folder");
    }

    List<FileSystemInfo> fileSystemInfos = new List<FileSystemInfo>();

    try
    {
        foreach (string filePath in Directory.GetFiles(folder, "*.*"))
        {
            // NOTE : We will add a FileSystemInfo object for each file found
            fileSystemInfos.Add(new FileInfo(filePath));
        }
    }
    catch
    {
        // NOTE : We are swallowing all exceptions here
        // Ideally they should be surfaced, and at least logged somewhere
        // Most of these will be security/permissions related, i.e.,
        // the Directory.GetFiles method will throw an exception if it
        // does not have security privileges to enumerate files in a folder.
    }
    try
    {
        foreach (string folderPath in Directory.GetDirectories(folder, "*"))
        {
            // NOTE : We will add a FileSystemInfo object for each directory found
            fileSystemInfos.Add(new DirectoryInfo(folderPath));

            // NOTE : We will also add all FileSystemInfo objects found under
            // each directory we find
            fileSystemInfos.AddRange(GetAllFilesAndFolders(folderPath));
        }
    }
    catch
    {
        // NOTE : We are swallowing all exceptions here
        // Ideally they should be surfaced, and at least logged somewhere
        // Most of these will be security/permissions related, i.e.,
        // the Directory.GetDirectories method will throw an exception if it
        // does not have security privileges to enumerate files in a folder.
    }

    return fileSystemInfos;
}

Следует отметить, что этот метод «обходит» всю структуру каталогов под папкой и НЕ возвращает, пока не «обходит» всю иерархию. Поэтому для возврата может потребоваться много времени, если найдено много объектов.

Следует также отметить, что читаемость этого метода может быть улучшена с помощью лямбда-выражений и методов расширения.

NB. Проблема с использованием Directory.GetFiles и Directory.GetDirectories для рекурсии подпапок заключается в том, что при возникновении каких-либо исключений (например, связанных с разрешениями безопасности) метод не возвращает ничего, тогда как рекурсивное выполнение вручную позволяет обработать его вручную. эти исключения и по-прежнему получить набор файлов обратно.

0 голосов
/ 10 октября 2009

Если вы хотите "удобный для начинающих метод", я могу предложить метод bobbymcr .

Но, если вы хотите сохранить структуру своего кода, я воспроизведу приведенный ниже код, который я разместил в ваш следующий вопрос :

static IEnumerable<FileSystemInfo> GetAllFilesAndDirectories(string path)
{
    string currentDirectory = "";
    string[] files = Directory.GetFiles( // skip empty subfolders
        path, "*.*", SearchOption.AllDirectories);
    foreach (string file in files)
    {
        if(currentDirectory != Path.GetDirectoryName(file))
        {
            // First time in this directory: return it
            currentDirectory = Path.GetDirectoryName(file);
            yield return new DirectoryInfo(currentDirectory);
        }

        yield return new FileInfo(file);
    }
}
0 голосов
/ 10 октября 2009

Вы специально просили "потушить это". Я думаю, что два других ответа очень хороши, но вот еще один способ сделать это с помощью еще более простого и понятного кода начального уровня.

http://support.microsoft.com/kb/303974

Редактировать

Я знаю, что это не 3.0, но это все еще проверенный, проверенный и простой для понимания способ сделать это.

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