C#でディレクトリ内のすべてのファイルを再帰的にリストする方法は?


315

ディレクトリ内のすべてのファイルとC#の子ディレクトリを再帰的にリストする方法は?


1
どこに入力しますか?if tree ...例は、dreamincode.net / code / snippet2591.htmです
Arsen Mkrtchyan

77
文字列[] filenames = Directory.GetFiles(path、 "*"、SearchOption.AllDirectories)
Bruce

あなたはを見てみたいことがあり、この質問、私が使用する再帰は、ツリービュー内のディレクトリ構造をレンダリングすることをコードサンプルを提示しています。ほとんどの場合、ロジックは同じです。
セレブラス2009年

5
これに関する問題は、単一のディレクトリにアクセスできない場合、簡単に壊れることです。結果はありません...
マークグラベル

1
一部のファイルにアクセスできないときに問題が発生した場合は、「ファイルを列挙して例外を
スローする

回答:


186

この記事はあなたが必要とするすべてをカバーします。ファイルを検索して名前を比較するのではなく、名前を印刷します。

次のように変更できます。

static void DirSearch(string sDir)
{
    try
    {
        foreach (string d in Directory.GetDirectories(sDir))
        {
            foreach (string f in Directory.GetFiles(d))
            {
                Console.WriteLine(f);
            }
            DirSearch(d);
        }
    }
    catch (System.Exception excpt)
    {
        Console.WriteLine(excpt.Message);
    }
}

barlopによって追加されました

GONealeは、上記では現在のディレクトリ内のファイルをリストしていないことを述べており、ディレクトリを取得する部分の外にファイルリストの部分を置くことを推奨しています。次のようにします。また、コメント解除できるWriteline行も含まれています。これは、再帰がどこにあるかを追跡するのに役立ち、再帰がどのように機能するかを示すための呼び出しを示すのに役立ちます。

            DirSearch_ex3("c:\\aaa");
            static void DirSearch_ex3(string sDir)
            {
                //Console.WriteLine("DirSearch..(" + sDir + ")");
                try
                {
                    Console.WriteLine(sDir);

                    foreach (string f in Directory.GetFiles(sDir))
                    {
                        Console.WriteLine(f);
                    }

                    foreach (string d in Directory.GetDirectories(sDir))
                    {
                        DirSearch_ex3(d);
                    }
                }
                catch (System.Exception excpt)
                {
                    Console.WriteLine(excpt.Message);
                }
            }

86
この方法では、初期ディレクトリのファイルはリストされません。サブディレクトリ以下のみがリストされます。私はGetFiles外GetDirectories移動する
GONeale

1
場合によっては、初期ディレクトリのファイルを必要としないこともあります。その場合、これはかなり小さな構造に最適です。:非常に大きなリスト、使用マルクGravellのソリューションのようなものについてはstackoverflow.com/a/929418/91189
ジョセフ・ガブリエル

2
@GONealeは正しいです。ユーザーが入力ルートディレクトリのファイルリストを期待しないことは、それほど妥当ではありません。ここでは単語入力が重要です。理由が入力されています。
Florin Mircea

2
内部のforeachループの周りにtry catchを追加する必要がありました。そうしないと、アクセス拒否エラーが継続しません
Shaun Vermaak '27

3
あなたは例外をキャッチすることを避けるべきです-たとえば、本当にOutOfMemoryExceptionをキャッチしたいですか?あなたが扱えるものだけを捕まえてください。
alastairtree 2018年

435

.NET 4.0には、(おそらく)イテレータベースの(配列ベースではなく)ファイル関数が組み込まれていることに注意してください。

foreach (string file in Directory.EnumerateFiles(path, "*.*", SearchOption.AllDirectories))
{
    Console.WriteLine(file);
}

現時点では、以下のようなものを使用します。組み込みの再帰メソッドは、単一のサブディレクトリにアクセスできない場合、簡単に壊れてしまいます...; Queue<string>使用方法は、あまりにも多くのコールスタック再帰を回避し、反復子ブロックは、私たちは巨大な配列を持つ回避できます。

static void Main() {
    foreach (string file in GetFiles(SOME_PATH)) {
        Console.WriteLine(file);
    }
}

static IEnumerable<string> GetFiles(string path) {
    Queue<string> queue = new Queue<string>();
    queue.Enqueue(path);
    while (queue.Count > 0) {
        path = queue.Dequeue();
        try {
            foreach (string subDir in Directory.GetDirectories(path)) {
                queue.Enqueue(subDir);
            }
        }
        catch(Exception ex) {
            Console.Error.WriteLine(ex);
        }
        string[] files = null;
        try {
            files = Directory.GetFiles(path);
        }
        catch (Exception ex) {
            Console.Error.WriteLine(ex);
        }
        if (files != null) {
            for(int i = 0 ; i < files.Length ; i++) {
                yield return files[i];
            }
        }
    }
}

1
@soandos再帰的な再ポイントでEnumerateFilesがIOExceptionをスローする「ファイルの名前はシステムで解決できません」
SerG

5
ファイル拡張子のないファイル*.*も含まれるかどうかを知りたいすべての人のために:はい、1分前にテストされました。
Tobias Knauss

1
これを使用するには、追加する必要がありますusing System.IO;
Reinstate Monica-Goodbye SE

7
@Wikiを使用Consoleするには、追加する必要があります。using System;ただし、IDEが必要なすべてのusingディレクティブ追加できるため(ctrl +。)、ここでは特別なものは何も使用していないため、それらを含めないのが一般的です。ヘック、あなたはまた、必要がありますclassだけでコトなどの定義を
マルクGravell

1
私たちは、そうステートメントを使用すると、常にでは歓迎さを含め、今.NETコアとVisual Studioのコードの世界にいる@MarcGravell 任意の検索や無意味な「ヤクシェービング」のシリーズを保存するための例.NETコード
JOHNC

98
Directory.GetFiles("C:\\", "*.*", SearchOption.AllDirectories)

2
ログインユーザーが一部のフォルダーにアクセスできない場合にエラーを回避する方法。
Romil Kumar Jain、2015

5
@Romil私は、このコードスニペットが完全な機能を示すのではなく、OPが求めていた生の機能のみを示しているとは思いません。共有してくれてありがとう、Pescuma!
kayleeFrye_onDeck 2015年

@kayleeFrye_onDeck、私はファイルの取得中にいずれかのフォルダーのレイズが発生した場合の心配のみを置きました。この懸念により、カスタム再帰関数を実装しています。
Romil Kumar Jain 2015

3
このソリューションでは「UnauthorizedAccessException」を受け取ります。このようなエラーを処理できるソリューションが必要です。
Kairan

13

.NET 4.5では、少なくとも、このバージョンはもっと短く、リストに含めるためのファイル基準を評価するというボーナスが追加されています。

public static IEnumerable<string> GetAllFiles(string path, 
                                              Func<FileInfo, bool> checkFile = null)
{
    string mask = Path.GetFileName(path);
    if (string.IsNullOrEmpty(mask)) mask = "*.*";
    path = Path.GetDirectoryName(path);
    string[] files = Directory.GetFiles(path, mask, SearchOption.AllDirectories);

    foreach (string file in files)
    {
        if (checkFile == null || checkFile(new FileInfo(file)))
            yield return file;
    }
}

次のように使用します。

var list = GetAllFiles(mask, (info) => Path.GetExtension(info.Name) == ".html").ToList();

これは、空のディレクトリがある場合を処理しません...関数内にreturnステートメントはありません。
FrumkinWY

@FrumkinWY空のディレクトリで何が起こりますか?これをテストするのに便利なマシンは今のところありません。
John Kaster、2018

12
IEnumerable<string> GetFilesFromDir(string dir) =>
 Directory.EnumerateFiles(dir).Concat(
 Directory.EnumerateDirectories(dir)
          .SelectMany(subdir => GetFilesFromDir(subdir)));

3

Framework 2.0で使用できます(ルートフォルダーのファイルを一覧表示します。最も一般的な答えが最善です):

static void DirSearch(string dir)
{
    try
    {
        foreach (string f in Directory.GetFiles(dir))
            Console.WriteLine(f);
        foreach (string d in Directory.GetDirectories(dir))
        {
            Console.WriteLine(d);
            DirSearch(d);
        }

    }
    catch (System.Exception ex)
    {
        Console.WriteLine(ex.Message);
    }
}

3

いくつかの優れた答えがありますが、これらの答えは私の問題を解決しませんでした。

フォルダーのアクセス許可の問題が発生するとすぐに: "アクセス許可が拒否されました"コードは失敗します。これは、「Permission Denied」の問題を回避するために使用したものです。

private int counter = 0;

    private string[] MyDirectories = Directory.GetDirectories("C:\\");

    private void ScanButton_Click(object sender, EventArgs e)
    {
        Thread MonitorSpeech = new Thread(() => ScanFiles());
        MonitorSpeech.Start();
    }

    private void ScanFiles()
    {
        string CurrentDirectory = string.Empty;

        while (counter < MyDirectories.Length)
        {
            try
            {
                GetDirectories();
                CurrentDirectory = MyDirectories[counter++];
            }
            catch
            {
                if (!this.IsDisposed)
                {
                    listBox1.Invoke((MethodInvoker)delegate { listBox1.Items.Add("Access Denied to : " + CurrentDirectory); });
                }
            }
        }
    }

    private void GetDirectories()
    {
        foreach (string directory in MyDirectories)
        {
            GetFiles(directory);
        }
    }

    private void GetFiles(string directory)
    {
        try
        {
            foreach (string file in Directory.GetFiles(directory, "*"))
            {
                listBox1.Invoke((MethodInvoker)delegate { listBox1.Items.Add(file); });
            }
        }
        catch
        {
            listBox1.Invoke((MethodInvoker)delegate { listBox1.Items.Add("Access Denied to : " + directory); });
        }
    }

これが他の人を助けることを願っています。


3

シンプルでクリーンなソリューション

/// <summary>
/// Scans a folder and all of its subfolders recursively, and updates the List of files
/// </summary>
/// <param name="sFullPath">Full path of the folder</param>
/// <param name="files">The list, where the output is expected</param>
internal static void EnumerateFiles(string sFullPath, List<FileInfo> fileInfoList)
{
    try
    {
        DirectoryInfo di = new DirectoryInfo(sFullPath);
        FileInfo[] files = di.GetFiles();

        foreach (FileInfo file in files)
            fileInfoList.Add(file);

        //Scan recursively
        DirectoryInfo[] dirs = di.GetDirectories();
        if (dirs == null || dirs.Length < 1)
            return;
        foreach (DirectoryInfo dir in dirs)
            EnumerateFiles(dir.FullName, fileInfoList);

    }
    catch (Exception ex)
    {
        Logger.Write("Exception in Helper.EnumerateFiles", ex);
    }
}

3
DirectoryInfo.GetFiles()がすぐにできることを手動で行っています-SearchOption.AllDirectoriesでオーバーロードを使用するだけで、すべてが単独で再帰します。これは複雑な解決策です。
philw

2

文字列だけでなくFileInfoも取得できるので、DirectoryInfoを使用することを好みます。

        string baseFolder = @"C:\temp";
        DirectoryInfo di = new DirectoryInfo(baseFolder);

        string searchPattern = "*.xml";

        ICollection<FileInfo> matchingFileInfos = di.GetFiles(searchPattern, SearchOption.AllDirectories)
            .Select(x => x)
            .ToList();

これは、将来FileInfoのプロパティに基づいて将来のフィルタリングが必要になる場合に備えて行います。

        string baseFolder = @"C:\temp";
        DirectoryInfo di = new DirectoryInfo(baseFolder);

        string searchPattern = "*.xml";

        ICollection<FileInfo> matchingFileInfos = di.GetFiles(searchPattern, SearchOption.AllDirectories)
            .Where(x => x.LastWriteTimeUtc < DateTimeOffset.Now)
            .Select(x => x)
            .ToList();

必要に応じて、弦に戻すこともできます。(そして、フィルター/ where-clauseについては、将来も保証されます。

        string baseFolder = @"C:\temp";
        DirectoryInfo di = new DirectoryInfo(baseFolder);

        string searchPattern = "*.xml";

        ICollection<string> matchingFileNames = di.GetFiles(searchPattern, SearchOption.AllDirectories)
            .Select(x => x.FullName)
            .ToList();

拡張子でファイラー化する場合は、「」が有効な検索パターンであることに注意してください。


1
private void GetFiles(DirectoryInfo dir, ref List<FileInfo> files)
{
    try
    {
        files.AddRange(dir.GetFiles());
        DirectoryInfo[] dirs = dir.GetDirectories();
        foreach (var d in dirs)
        {
            GetFiles(d, ref files);
        }
    }
    catch (Exception e)
    {

    }
}

1
なぜパラメータfilesref?必要はありません。
Massimiliano Kraus

@MassimilianoKraus必須ではありませんが、それによって彼のメソッドが変更されfilesnew List<FileInfo>()役に立たないパラメーターとして渡すことはできなくなることが明確になります。必要に応じて、ある程度の最適化を可能にし、新しいオブジェクトの作成を回避することができます。
jeromej

@JeromeJ OOPがわかっている場合は、オブジェクトをメソッドに渡すたびに、そのメソッドがオブジェクトのプロパティ/フィールドを変更できることがわかります。したがってref、何も明確になりません。ref目的は、全体変更することでfilesも、メソッドの呼び出し元のためのポインタを:それは危険な操作だとここにいることは必要ありません:あなただけのリストを埋めることができますが、あなたは上の別のリストに再ポイントにそれを必要としませんヒープ。ref非常に特殊な場合にのみ使用してください。ほとんどの場合、より機能的なパラダイムの方法で実装する必要があります。
Massimiliano Kraus

1

を避けるためにUnauthorizedAccessException、私は以下を使用します:

var files = GetFiles(@"C:\", "*.*", SearchOption.AllDirectories);
foreach (var file in files)
{
    Console.WriteLine($"{file}");
}

public static IEnumerable<string> GetFiles(string path, string searchPattern, SearchOption searchOption)
{
    var foldersToProcess = new List<string>()
    {
        path
    };

    while (foldersToProcess.Count > 0)
    {
        string folder = foldersToProcess[0];
        foldersToProcess.RemoveAt(0);

        if (searchOption.HasFlag(SearchOption.AllDirectories))
        {
            //get subfolders
            try
            {
                var subfolders = Directory.GetDirectories(folder);
                foldersToProcess.AddRange(subfolders);
            }
            catch (Exception ex)
            {
                //log if you're interested
            }
        }

        //get files
        var files = new List<string>();
        try
        {
            files = Directory.GetFiles(folder, searchPattern, SearchOption.TopDirectoryOnly).ToList();
        }
        catch (Exception ex)
        {
            //log if you're interested
        }

        foreach (var file in files)
        {
            yield return file;
        }
    }
}

1

ファイル名だけが必要で、私がここでのほとんどの解決策(機能的または読みやすさ的)を本当に好きではなかったので、この怠惰な方法はどうですか?

private void Foo()
{
  var files = GetAllFiles("pathToADirectory");
  foreach (string file in files)
  {
      // Use can use Path.GetFileName() or similar to extract just the filename if needed
      // You can break early and it won't still browse your whole disk since it's a lazy one
  }
}

/// <exception cref="T:System.IO.DirectoryNotFoundException">The specified path is invalid (for example, it is on an unmapped drive).</exception>
/// <exception cref="T:System.UnauthorizedAccessException">The caller does not have the required permission.</exception>
/// <exception cref="T:System.IO.IOException"><paramref name="path" /> is a file name.-or-A network error has occurred.</exception>
/// <exception cref="T:System.IO.PathTooLongException">The specified path, file name, or both exceed the system-defined maximum length. For example, on Windows-based platforms, paths must be less than 248 characters and file names must be less than 260 characters.</exception>
/// <exception cref="T:System.ArgumentNullException"><paramref name="path" /> is null.</exception>
/// <exception cref="T:System.ArgumentException"><paramref name="path" /> is a zero-length string, contains only white space, or contains one or more invalid characters as defined by <see cref="F:System.IO.Path.InvalidPathChars" />.</exception>
[NotNull]
public static IEnumerable<string> GetAllFiles([NotNull] string directory)
{
  foreach (string file in Directory.GetFiles(directory))
  {
    yield return file; // includes the path
  }

  foreach (string subDir in Directory.GetDirectories(directory))
  {
    foreach (string subFile in GetAllFiles(subDir))
    {
      yield return subFile;
    }
  }
}


0

ここに、Hernaldoのものに基づいた私の見解を示します。名前のどこかに特定の文字列が含まれているXMLファイルなど、特定のパターンの名前を持つファイルを見つける必要がある場合は、次のようになります。

// call this like so: GetXMLFiles("Platypus", "C:\\");
public static List<string> GetXMLFiles(string fileType, string dir)
{
    string dirName = dir; 
    var fileNames = new List<String>();
    try
    {
        foreach (string f in Directory.GetFiles(dirName))
        {
            if ((f.Contains(fileType)) && (f.Contains(".XML")))
            {
                fileNames.Add(f);
            }
        }
        foreach (string d in Directory.GetDirectories(dirName))
        {
            GetXMLFiles(fileType, d);
        }
    }
    catch (Exception ex)
    {
        MessageBox.Show(ex.Message);
    }
    return fileNames;
}

0

モデル化するファイルとフォルダーのリスト、カスタム実装。
これにより、開始ディレクトリから始まるすべてのファイルとフォルダの完全なリストが作成されます。

public class DirOrFileModel
    {
        #region Private Members

        private string _name;
        private string _location;
        private EntryType _entryType;

        #endregion

        #region Bindings

        public string Name
        {
            get { return _name; }
            set
            {
                if (value == _name) return;
                _name = value;
            }
        }

        public string Location
        {
            get { return _location; }
            set
            {
                if (value == _location) return;
                _location = value;
            }
        }

        public EntryType EntryType
        {
            get { return _entryType; }
            set
            {
                if (value == _entryType) return;
                _entryType = value;
            }
        }

        public ObservableCollection<DirOrFileModel> Entries { get; set; }

        #endregion

        #region Constructor

        public DirOrFileModel()
        {
            Entries = new ObservableCollection<DirOrFileModel>();
        }

        #endregion
    }

    public enum EntryType
    {
        Directory = 0,
        File = 1
    }

方法:

 static DirOrFileModel DirSearch(DirOrFileModel startDir)
        {
            var currentDir = startDir;
            try
            {
                foreach (string d in Directory.GetDirectories(currentDir.Location))
                {
                    var newDir = new DirOrFileModel
                    {
                        EntryType = EntryType.Directory,
                        Location = d,
                        Name = Path.GetFileName(d)
                    };
                    currentDir.Entries.Add(newDir);

                    DirSearch(newDir);
                }

                foreach (string f in Directory.GetFiles(currentDir.Location))
                {
                    var newFile = new DirOrFileModel
                    {
                        EntryType = EntryType.File,
                        Location = f,
                        Name = Path.GetFileNameWithoutExtension(f)
                    };
                    currentDir.Entries.Add(newFile);
                }

            }
            catch (Exception excpt)
            {
                Console.WriteLine(excpt.Message);
            }
            return startDir;
        }

使用法:

var dir = new DirOrFileModel
            {
                Name = "C",
                Location = @"C:\",
                EntryType = EntryType.Directory
            };

            dir = DirSearch(dir);

0

短くてシンプルなソリューション

string dir = @"D:\PATH";

DateTime from_date = DateTime.Now.Date;
DateTime to_date = DateTime.Now.Date.AddHours(23);
var files = Directory.EnumerateFiles(dir, "*.*",SearchOption.AllDirectories).Select(i=>new FileInfo(i))
.Where(file=>file.LastWriteTime >= from_date && file.LastWriteTime <= to_date);
foreach(var fl in files)
    Console.WriteLine(fl.FullName);

0

これは私がディレクトリとサブディレクトリ内のすべてのファイルを取得するのに役立ちました、誰かのために役立つかもしれません。[上記の回答から発想を得た]

static void Main(string[] args)
    {
        try
        {
            var root = @"G:\logs";
            DirectorySearch(root);
        }
        catch (Exception ex)
        {
            Console.WriteLine(ex.Message);
        }
        Console.ReadKey();
    }





public static void DirectorySearch(string root, bool isRootItrated = false)
{
    if (!isRootItrated)
    {
        var rootDirectoryFiles = Directory.GetFiles(root);
        foreach (var file in rootDirectoryFiles)
        {
            Console.WriteLine(file);
        } 
    }

    var subDirectories = Directory.GetDirectories(root);
    if (subDirectories?.Any() == true)
    {
        foreach (var directory in subDirectories)
        {
            var files = Directory.GetFiles(directory);
            foreach (var file in files)
            {
                Console.WriteLine(file);
            }
            DirectorySearch(directory, true);
        }
    }
}

0
var d = new DirectoryInfo(@"C:\logs");
var list = d.GetFiles("*.txt").Select(m => m.Name).ToList();

0

ディレクトリとフォルダーを除外するオプションに移動するためのmax lvlを含むいくつかの改善されたバージョン:

using System;
using System.IO;

class MainClass {
  public static void Main (string[] args) {

    var dir = @"C:\directory\to\print";
    PrintDirectoryTree(dir, 2, new string[] {"folder3"});
  }


  public static void PrintDirectoryTree(string directory, int lvl, string[] excludedFolders = null, string lvlSeperator = "")
  {
    excludedFolders = excludedFolders ?? new string[0];

    foreach (string f in Directory.GetFiles(directory))
    {
        Console.WriteLine(lvlSeperator+Path.GetFileName(f));
    } 

    foreach (string d in Directory.GetDirectories(directory))
    {
        Console.WriteLine(lvlSeperator + "-" + Path.GetFileName(d));

        if(lvl > 0 && Array.IndexOf(excludedFolders, Path.GetFileName(d)) < 0)
        {
          PrintDirectoryTree(d, lvl-1, excludedFolders, lvlSeperator+"  ");
        }
    }
  }
}

入力ディレクトリ:

-folder1
  file1.txt
  -folder2
    file2.txt
    -folder5
      file6.txt
  -folder3
    file3.txt
  -folder4
    file4.txt
    file5.txt

関数の出力(lvlの制限によりfolder5のコンテンツは除外され、excludeFolders配列にあるためfolder3のコンテンツは除外されます):

-folder1
  file1.txt
  -folder2
    file2.txt
    -folder5
  -folder3
  -folder4
    file4.txt
    file5.txt

-1

これは、B。クレイシャノンのコードのバージョンで、Excelファイルでは静的ではありません。

class ExcelSearcher
{
    private List<string> _fileNames;

    public ExcelSearcher(List<string> filenames)
    {
        _fileNames = filenames;
    }
    public List<string> GetExcelFiles(string dir, List<string> filenames = null)
    {

        string dirName = dir;
        var dirNames = new List<string>();
        if (filenames != null)
        {
            _fileNames.Concat(filenames);
        }
        try
        {
            foreach (string f in Directory.GetFiles(dirName))
            {
                if (f.ToLower().EndsWith(".xls") || f.ToLower().EndsWith(".xlsx"))
                {
                    _fileNames.Add(f);
                }
            }
            dirNames = Directory.GetDirectories(dirName).ToList();
            foreach (string d in dirNames)
            {
                GetExcelFiles(d, _fileNames);
            }
        }
        catch (Exception ex)
        {
            //Bam
        }
        return _fileNames;
    }

-1

非常にシンプルなソリューションで、ファイルのリストを返します。

    public static List<string> AllFilesInFolder(string folder)
    {
        var result = new List<string>();

        foreach (string f in Directory.GetFiles(folder))
        {
            result.Add(f);
        }

        foreach (string d in Directory.GetDirectories(folder))
        {
            result.AddRange(AllFilesInFolder(d));
        }

        return result;
    }

-2
static void Main(string[] args)
        {
            string[] array1 = Directory.GetFiles(@"D:\");
            string[] array2 = System.IO.Directory.GetDirectories(@"D:\");
            Console.WriteLine("--- Files: ---");
            foreach (string name in array1)
            {
                Console.WriteLine(name);
            }
            foreach (string name in array2)
            {
                Console.WriteLine(name);
            }
                  Console.ReadLine();
        }

1
ええと...これは再帰的ではありません
mxmissile 2018
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.