ディレクトリ+サブディレクトリ内のすべてのファイルとディレクトリをリストする


109

ディレクトリとそのディレクトリのサブディレクトリに含まれるすべてのファイルとディレクトリを一覧表示したいのですが。ディレクトリとしてC:\を選択した場合、プログラムは、アクセス可能なハードドライブ上のすべてのファイルとフォルダーのすべての名前を取得します。

リストは次のようになります

fd \ 1.txt
fd \ 2.txt
fd \ a \
fd \ b \
fd \ a \ 1.txt
fd \ a \ 2.txt
fd \ a \ a \
fd \ a \ b \
fd \ b \ 1.txt
fd \ b \ 2.txt
fd \ b \ a
fd \ b \ b
fd \ a \ a \ 1.txt
fd \ a \ a \ a \
fd \ a \ b \ 1.txt
fd \ a \ b \ a
fd \ b \ a \ 1.txt
fd \ b \ a \ a \
fd \ b \ b \ 1.txt
fd \ b \ b \ a

System.IO名前空間を参照して、役立つと思われるクラスメソッドを探します。
Lucero、

この質問を確認して、パターンに一致する部分を削除してください。
dasblinkenlight 2012

回答:


192
string[] allfiles = Directory.GetFiles("path/to/dir", "*.*", SearchOption.AllDirectories);

*.*ファイルを照合するパターンはどこですか

ディレクトリも必要な場合は、次のようにできます。

 foreach (var file in allfiles){
     FileInfo info = new FileInfo(file);
 // Do something with the Folder or just add them to a list via nameoflist.add();
 }

1
本当に働かない… Lsit<>クラス?GetFilesは何を返しますか?また、要求されたディレクトリ名はどうですか?
Lucero、

1
このGetFilesメソッドは文字列配列を返します。
Guffa

実際...あなたは正しい...私は2日前にQtアバウトを学んでいて、少し間違えられました
Ruslan F.

これはうまくいくかもしれませんが、UnauthorizedAccessExceptionで失敗することがよくあります。アクセスできるディレクトリのみをどのように検索しますか?
derp_in_mouth 2012

これは、システムでこのアプリに十分な権限がないことを意味します
Ruslan F.

50

Directory.GetFileSystemEntries.NET 4.0以降に存在し、ファイルとディレクトリの両方を返します。次のように呼び出します。

string[] entries = Directory.GetFileSystemEntries(path, "*", SearchOption.AllDirectories);

(UnauthorizedAccessException)にアクセスできないサブディレクトリの内容を一覧表示する試みには対応しませんが、ニーズには十分であることに注意してください。


3
これは断然最良の答えです。すべてのファイルとフォルダーを1行のコードで取得しますが、他のファイルでは実行できません。
スティーブスミス

15

GetDirectoriesおよびGetFilesメソッドを使用して、フォルダとファイルを取得します。

を使用して、サブフォルダー内のフォルダーとファイルも取得します。SearchOption AllDirectories


サブストリングを使用して、名前の左側の部分を切り取ります。:)
Lucero

@Luceroどのようにそしてなぜあなたはそれをしますか?Pathより信頼できる方法を提供します。
Gusdor

@Gusdor Pathパスの左側の固定部分を削除するためのを使用して、より適切な方法を提案してください。たとえば、上記の例では「C:」です。
Lucero

@Lucero私のコメントは不適切な言い回しでした。「部分文字列を使用する」は多くのことを教えてくれず、素晴らしい解決策を導き出すためにlinqpadにこだわる必要がありました。たとえば、パラメータは何でしょうか?path.SubString(2)ドライブ文字とコロンを単純に削除するつもりですか?ディレクトリがネットワーク共有の場合はどうなりますか?Path信頼できる方法としては、この領域で多くの優れた機能を提供できるため、お勧めします。この場合、あなたは書くかもしれませんfilePath.Substring(Path.GetPathRoot(filePath).Length)。はい、これは最も簡潔なのでSubstringを使用します。
Gusdor

10
public static void DirectorySearch(string dir)
{
    try
    {
        foreach (string f in Directory.GetFiles(dir))
        {
            Console.WriteLine(Path.GetFileName(f));
        }
        foreach (string d in Directory.GetDirectories(dir))
        {
            Console.WriteLine(Path.GetFileName(d));
            DirectorySearch(d);
        }
    }
    catch (System.Exception ex)
    {
        Console.WriteLine(ex.Message);
    }
}

3
コードが何をするのかについて少し説明を追加できれば、回答が改善されます。
Alex

ディレクトリを再帰的に処理し、ファイル名またはディレクトリ名を出力します。すべての内部ディレクトリについて、同じ関数を呼び出します。:詳細については stackoverflow.com/questions/929276/...
I.Step

3

申し訳ありGetFilesませんが、メソッドはファイルのリストを返しますが、ディレクトリは返しません。質問のリストから、結果にはフォルダも含める必要があることがわかります。よりカスタマイズされたリストが必要な場合はGetFilesGetDirectories再帰的に呼び出してみてください。これを試して:

List<string> AllFiles = new List<string>();
void ParsePath(string path)
{
    string[] SubDirs = Directory.GetDirectories(path);
    AllFiles.AddRange(SubDirs);
    AllFiles.AddRange(Directory.GetFiles(path));
    foreach (string subdir in SubDirs)
        ParsePath(subdir);
}

ヒント:特定の属性を確認する必要がある場合はFileInfo、およびDirectoryInfoクラスを使用できます。


1

ハンドルを返すFindFirstFileを使用してから、FindNextFileを呼び出す関数を再帰的に呼び出すことができます。これは、参照される構造体がAlternativeName、lastTmeCreated、modified、attributesなどのさまざまなデータで満たされるため、これは良いアプローチです。

ただし、.netフレームワークを使用する場合は、管理されていない領域に入る必要があります。


1

ディレクトリとフォルダーを除外するオプションに移動する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

0

ディレクトリツリー内のサブフォルダーにアクセスできない場合、Directory.GetFilesは停止し、例外をスローして、受信文字列[]にnull値を生成します。

ここでは、この回答を参照して くださいhttps://stackoverflow.com/a/38959208/6310707

ループ内の例外を管理し、フォルダー全体がトラバースされるまで機能し続けます。


0

論理的で順序付けられた方法:

using System;
using System.Collections.Generic;
using System.IO;
using System.Reflection;

namespace DirLister
{
class Program
{
    public static void Main(string[] args)
    {
        //with reflection I get the directory from where this program is running, thus listing all files from there and all subdirectories
        string[] st = FindFileDir(Path.GetDirectoryName(Assembly.GetEntryAssembly().Location));
        using ( StreamWriter sw = new StreamWriter("listing.txt", false ) )
        {
            foreach(string s in st)
            {
                //I write what I found in a text file
                sw.WriteLine(s);
            }
        }
    }

    private static string[] FindFileDir(string beginpath)
    {
        List<string> findlist = new List<string>();

        /* I begin a recursion, following the order:
         * - Insert all the files in the current directory with the recursion
         * - Insert all subdirectories in the list and rebegin the recursion from there until the end
         */
        RecurseFind( beginpath, findlist );

        return findlist.ToArray();
    }

    private static void RecurseFind( string path, List<string> list )
    {
        string[] fl = Directory.GetFiles(path);
        string[] dl = Directory.GetDirectories(path);
        if ( fl.Length>0 || dl.Length>0 )
        {
            //I begin with the files, and store all of them in the list
            foreach(string s in fl)
                list.Add(s);
            //I then add the directory and recurse that directory, the process will repeat until there are no more files and directories to recurse
            foreach(string s in dl)
            {
                list.Add(s);
                RecurseFind(s, list);
            }
        }
    }
}
}

あなたのコードが何をしているのか、説明やインラインコメントを提供していただけませんか?
MarthyM 2017年

もちろん、それを行いましたが、それは自明であるはずです。それは、すべてのディレクトリとファイルを通る単純なループ再帰です
Sascha

0

次の例は最速です例外を処理するディレクトリツリー内のファイルとサブフォルダーをリストする(並列化されていない)方法です。SearchOption.AllDirectoriesを使用してDirectory.EnumerateDirectoriesを使用してすべてのディレクトリを列挙する方が高速ですが、UnauthorizedAccessExceptionまたはPathTooLongExceptionにヒットすると、このメソッドは失敗します。

総称スタックコレクションタイプを使用します。これは後入れ先出し(LIFO)スタックであり、再帰を使用しません。https://msdn.microsoft.com/en-us/library/bb513869.aspxから、すべてのサブディレクトリとファイルを列挙し、それらの例外を効果的に処理できます。

    public class StackBasedIteration
{
    static void Main(string[] args)
    {
        // Specify the starting folder on the command line, or in 
        // Visual Studio in the Project > Properties > Debug pane.
        TraverseTree(args[0]);

        Console.WriteLine("Press any key");
        Console.ReadKey();
    }

    public static void TraverseTree(string root)
    {
        // Data structure to hold names of subfolders to be
        // examined for files.
        Stack<string> dirs = new Stack<string>(20);

        if (!System.IO.Directory.Exists(root))
        {
            throw new ArgumentException();
        }
        dirs.Push(root);

        while (dirs.Count > 0)
        {
            string currentDir = dirs.Pop();
            string[] subDirs;
            try
            {
                subDirs = System.IO.Directory.EnumerateDirectories(currentDir); //TopDirectoryOnly
            }
            // An UnauthorizedAccessException exception will be thrown if we do not have
            // discovery permission on a folder or file. It may or may not be acceptable 
            // to ignore the exception and continue enumerating the remaining files and 
            // folders. It is also possible (but unlikely) that a DirectoryNotFound exception 
            // will be raised. This will happen if currentDir has been deleted by
            // another application or thread after our call to Directory.Exists. The 
            // choice of which exceptions to catch depends entirely on the specific task 
            // you are intending to perform and also on how much you know with certainty 
            // about the systems on which this code will run.
            catch (UnauthorizedAccessException e)
            {                    
                Console.WriteLine(e.Message);
                continue;
            }
            catch (System.IO.DirectoryNotFoundException e)
            {
                Console.WriteLine(e.Message);
                continue;
            }

            string[] files = null;
            try
            {
                files = System.IO.Directory.EnumerateFiles(currentDir);
            }

            catch (UnauthorizedAccessException e)
            {

                Console.WriteLine(e.Message);
                continue;
            }

            catch (System.IO.DirectoryNotFoundException e)
            {
                Console.WriteLine(e.Message);
                continue;
            }
            // Perform the required action on each file here.
            // Modify this block to perform your required task.
            foreach (string file in files)
            {
                try
                {
                    // Perform whatever action is required in your scenario.
                    System.IO.FileInfo fi = new System.IO.FileInfo(file);
                    Console.WriteLine("{0}: {1}, {2}", fi.Name, fi.Length, fi.CreationTime);
                }
                catch (System.IO.FileNotFoundException e)
                {
                    // If file was deleted by a separate application
                    //  or thread since the call to TraverseTree()
                    // then just continue.
                    Console.WriteLine(e.Message);
                    continue;
                }
                catch (UnauthorizedAccessException e)
                {                    
                    Console.WriteLine(e.Message);
                    continue;
                }
            }

            // Push the subdirectories onto the stack for traversal.
            // This could also be done before handing the files.
            foreach (string str in subDirs)
                dirs.Push(str);
        }
    }
}

膨大な数のファイルやディレクトリにタスクを使用していますか?
PreguntonCojoneroCabrón

msdn.microsoft.com/en-us/library/ff477033(v=vs.110).aspxは、スタックコレクションを使用した上記のソリューションの並列スレッディングバージョンであり、高速です。
マーカス

0

次のコードを2つのボタンがあるフォームで使用します。1つは終了用、もう1つは開始用です。フォルダブラウザダイアログとファイル保存ダイアログ。コードは以下にリストされており、私のシステムで動作しますWindows10(64):

using System;
using System.IO;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;

namespace Directory_List
{

    public partial class Form1 : Form
    {
        public string MyPath = "";
        public string MyFileName = "";
        public string str = "";

        public Form1()
        {
            InitializeComponent();
        }    
        private void cmdQuit_Click(object sender, EventArgs e)
        {
            Application.Exit();
        }    
        private void cmdGetDirectory_Click(object sender, EventArgs e)
        {
            folderBrowserDialog1.ShowDialog();
            MyPath = folderBrowserDialog1.SelectedPath;    
            saveFileDialog1.ShowDialog();
            MyFileName = saveFileDialog1.FileName;    
            str = "Folder = " + MyPath + "\r\n\r\n\r\n";    
            DirectorySearch(MyPath);    
            var result = MessageBox.Show("Directory saved to Disk!", "", MessageBoxButtons.OK);
                Application.Exit();    
        }    
        public void DirectorySearch(string dir)
        {
                try
            {
                foreach (string f in Directory.GetFiles(dir))
                {
                    str = str + dir + "\\" + (Path.GetFileName(f)) + "\r\n";
                }    
                foreach (string d in Directory.GetDirectories(dir, "*"))
                {

                    DirectorySearch(d);
                }
                        System.IO.File.WriteAllText(MyFileName, str);

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

-1
using System.IO;
using System.Text;
string[] filePaths = Directory.GetFiles(@"path", "*.*", SearchOption.AllDirectories);

あなたの回答は、既存のトップ投票の回答に新しいものを追加するものではありません。
デフォルトロケール

1
これも間違っています。これは、(指定された質問として)ディレクトリを返さず、実際のファイルのみを返すためです。
Alastair Maw

-1

littlebitはシンプルでゆっくりと動作します。ファイルパスを指定しない場合、基本的には「fixPath」を使用します。これは単なる例です。目的の正しいファイルタイプを検索できます。「temporaryFileListは検索されたファイルリストなので、リスト名を選択したときに間違いがありました。だからそれを続けて....そして「errorList」はそれ自体が語っています

 static public void Search(string path, string fileType, List<string> temporaryFileList, List<string> errorList)
    {

        List<string> temporaryDirectories = new List<string>();

        //string fix = @"C:\Users\" + Environment.UserName + @"\";
        string fix = @"C:\";
        string folders = "";
        //Alap útvonal megadása 
        if (path.Length != 0)
        { folders = path; }
        else { path = fix; }

        int j = 0;
        int equals = 0;
        bool end = true;

        do
        {

            equals = j;
            int k = 0;

            try
            {

                int foldersNumber = 
                Directory.GetDirectories(folders).Count();
                int fileNumber = Directory.GetFiles(folders).Count();

                if ((foldersNumber != 0 || fileNumber != 0) && equals == j)
                {

                    for (int i = k; k < 
                    Directory.GetDirectories(folders).Length;)
                    {

             temporaryDirectories.Add(Directory.GetDirectories(folders)[k]);
                        k++;
                    }

                    if (temporaryDirectories.Count == j)
                    {
                        end = false;
                        break;
                    }
                    foreach (string files in Directory.GetFiles(folders))
                    {
                        if (files != string.Empty)
                        {
                            if (fileType.Length == 0)
                            {
                                temporaryDirectories.Add(files);
                            }
                            else
                            {

                                if (files.Contains(fileType))
                                {
                                    temporaryDirectories.Add(files);

                                }
                            }
                        }
                        else
                        {
                            break;
                        }
                    }

                }

                equals++;

                for (int i = j; i < temporaryDirectories.Count;)
                {
                    folders = temporaryDirectories[i];
                    j++;
                    break;
                }

            }
            catch (Exception ex)
            {
                errorList.Add(folders);

                for (int i = j; i < temporaryDirectories.Count;)
                {
                    folders = temporaryDirectories[i];
                    j++;
                    break;
                }
            }
        } while (end);
    }

-1

文字列のリストを作成

    public static List<string> HTMLFiles = new List<string>();

 private void Form1_Load(object sender, EventArgs e)
        {

     HTMLFiles.AddRange(Directory.GetFiles(@"C:\DataBase", "*.txt"));
            foreach (var item in HTMLFiles)
            {
                MessageBox.Show(item);
            }

}

これはサブディレクトリを取得しません。
TidyDev

-1

dir / s / b > results.txt

/ s =サブフォルダ/ b =クマの結果

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