3
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

【C#】【ファイル検索高速化】ファイル検索が遅いと感じる方へ!高速でファイル検索を行うファイル検索ツールをC#で作ってみた

Posted at

Windowsの検索バーで行うファイル検索は、遅いと感じたことはありませんか?

無題.png

それもそのはず、ファイルは階層構造になっていて、検索するには奥深くまで全て調べなければいけません。

そこで、ファイル検索を非同期で並列検索して高速化できないか考えました。

Taskクラスの使用

ファイルシステムなどツリー構造になっているものはプログラム上で「再帰呼び出し」という方法を使って探索します。

そこで、Taskクラスを使用して再帰呼び出しの部分を非同期で行うことにしました。

以下にコードを示します。

FrmMain.cs
        /// <summary>
        ///     ファイル検索を行う。
        /// </summary>
        /// <param name="strFileName">ファイル名</param>
        /// <param name="strDirOrFile">現在探索中のディレクトリまたはファイル</param>
        /// <param name="blnIgnoreDir">フォルダパスを無視する</param>
        /// <param name="kind">ファイルまたはフォルダ</param>
        private void SearchFile(string strFileName, string strDirOrFile, bool blnIgnoreDir, FileFolder kind)
        {
            if (this.blnCancel) return;

            Action action = () =>
            {
                Interlocked.Increment(ref this.intRunCount);
                try
                {
                    if (kind == FileFolder.File)
                    {
                        string strTarget = strDirOrFile;
                        if (blnIgnoreDir) strTarget = System.IO.Path.GetFileName(strTarget);
                        if (this.IsMatch(strFileName, strTarget))
                        {
                            this.context.Post(d =>
                            {
                                DgvRowFilePath rowBound = new DgvRowFilePath() { FilePath = strDirOrFile };
                                this.lstList.Add(rowBound);
                            }, null);
                        }
                        Interlocked.Increment(ref this.intFileCount);
                    }
                    else
                    {
                        this.context.Post(d =>
                        {
                            if (this.blnCancel) return;
                            this.lblStatus.Text = this.intFileCount + "ファイル検索済み 探索中:" + strDirOrFile.Replace(this.txtBaseDir.Text + "\\", "");
                            this.lblTime.Text = (this.sw.ElapsedMilliseconds / 1000) + "秒経過";
                        }, null);
                        string[] aryFiles = new string[0];
                        string[] aryFolders = new string[0];
                        try { aryFiles = System.IO.Directory.GetFiles(strDirOrFile); } catch (Exception) { }
                        try { aryFolders = System.IO.Directory.GetDirectories(strDirOrFile); } catch (Exception) { }
                        foreach (string file in aryFiles) this.SearchFile(strFileName, file, blnIgnoreDir, FileFolder.File);
                        foreach (string folder in aryFolders) this.SearchFile(strFileName, folder, blnIgnoreDir, FileFolder.Folder);
                    }
                }
                finally
                {
                    Interlocked.Decrement(ref this.intRunCount);
                }
            };

            Task.Run(action);
        }

ファイル検索のメソッド自体を非同期処理の対象とします。ディレクトリがあればそのディレクトリ内を別のスレッドで探索処理します。さらにその中でディレクトリがあればさらに別のスレッドで処理を行います。

見つかったファイル数this.intFileCountは非同期の場合インクリメント(++)するとスレッドセーフにならないため、System.Threading名前空間のInterlocked.Incrementメソッドを使用してインクリメントします。

このプログラムではスレッドの実行数this.intRunCountを保持しており、簡易的な処理としてスレッドの実行数が0になったタイミングで検索結果を表示するようにしています。

最後に

ファイル検索が遅く悩んでいる方が、本記事により、お役立ちいただければ幸いです。
また、次のリンクにてソースを公開しておりますので、参考にして改良していただければ幸いです。ソース自体は自由に使用して頂いて構いません。

GitHubのリンクはこちらになります。
https://github.com/koshiRyu/FileExplorer

最後まで見てくださりありがとうございます。

3
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?