0
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.

【iOS】【Objective-C】【Swift】指定したディレクトリにあるファイルの一覧を取得する方法

Last updated at Posted at 2023-02-24

はじめに

Swiftで、ドキュメントの一覧を出す方法ってなんかあるか調べてみました。

指定したディレクトリにあるファイルの一覧を取得する方法

 iOSのiOSのSwiftで指定したディレクトリにあるファイルの一覧を取得する方法については、以下のように簡潔にまとめることができます。

指定したディレクトリにあるファイルの一覧を取得する方法 例
if let documentsURL = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).first {
    do {
        let fileURLs = try FileManager.default.contentsOfDirectory(at: documentsURL, includingPropertiesForKeys: nil)
        // <#取得したファイル一覧を利用する処理を記述#>
    } catch {
        print("Error while enumerating files \(documentsURL.path): \(error.localizedDescription)")
    }
}

 このコードでは、FileManagerクラスのcontentsOfDirectoryメソッドを使用して、指定されたディレクトリに含まれるファイル一覧を取得しています。このメソッドは、指定されたディレクトリ内の浅い検索を行い、含まれている項目のURLの配列を返します。

 例えば、上記のコードでは、.documentDirectory(iOSアプリのドキュメントディレクトリ)を指定しています。このディレクトリに含まれるファイル一覧が取得され、fileURLsに代入されます。取得したファイル一覧を利用する処理は、コメントで記述されています。

try-catch文 による例外処理の対応

上記のコードでは、try-catch文を使用して、エラーハンドリングを行っています。tryブロック内でエラーが発生した場合には、catchブロックでエラーを受け取り、適切な処理を行います。

指定したディレクトリにあるファイルの一覧を取得する方法 例 catch の抜粋
} catch {
    print("Error while enumerating files \(documentsURL.path): \(error.localizedDescription)")
}

エラーを受け取るために、catchブロックの引数にError型のerrorという名前を設定しています。このerrorは、tryブロックで発生したエラーの情報を持っています。ここでは、エラーが発生した場合には、エラーメッセージを出力するようにしています。

例外処理を推奨しているので、何らかのエラーが発生した場合にはエラーメッセージが出力しておくようにしておくとエラーが発生した時に何のエラーが発生しているかがわかります。

こちらのコードは、Swift 4以降で使用可能です。また、深い列挙を行う場合には、enumeratorメソッドを使用する必要があります。

enumeratorメソッド

enumeratorメソッドは、FileManagerクラスに定義されているメソッドで、指定されたディレクトリ以下に含まれるファイルやディレクトリを再帰的に取得することができます。contentsOfDirectoryメソッドと異なり、深い列挙を行うことができるため、サブディレクトリ内に含まれるファイルやディレクトリも取得することができます。

enumeratorメソッドは、URLを引数に取り、URLの配列を返します。以下は、enumeratorメソッドを使用して、指定されたディレクトリ以下のすべてのファイルとディレクトリを取得する例です。

Swift 4でenumeratorメソッドを使用して、指定されたディレクトリ以下のすべてのファイルとディレクトリを取得する例
if let documentDirectory = FileManager.default.urls(for: .documentDirectory, in: .userDomainMask).first {
    let fileManager = FileManager.default
    let keys = [URLResourceKey.isDirectoryKey, URLResourceKey.localizedNameKey]
    let enumerator = fileManager.enumerator(at: documentDirectory, includingPropertiesForKeys: keys)

    while let url = enumerator?.nextObject() as? URL {
        do {
            let resourceValues = try url.resourceValues(forKeys: Set(keys))
            if resourceValues.isDirectory == true {
                print("\(url.path) is a directory")
            } else {
                print("\(url.path) is a file")
            }
        } catch {
            print("Error: \(error.localizedDescription)")
        }
    }
}

 上記のコードでは、documentDirectoryに指定されたディレクトリ以下に含まれるすべてのファイルとディレクトリを取得しています。enumeratorメソッドにより、ファイルのURLの配列を取得し、whileループで順次アクセスしています。取得したURLに対して、resourceValuesメソッドを使用して、URLの情報を取得しています。この情報には、isDirectoryKeyやlocalizedNameKeyなどが含まれており、ディレクトリであるかどうかなどを判断するために使用されます。

 enumeratorメソッドは、contentsOfDirectoryメソッドと異なり、深い列挙を行うことができるため、サブディレクトリ内に含まれるファイルやディレクトリも取得することができます。ただし、再帰的にアクセスするため、パフォーマンスに影響がある場合がありますので、必要最小限の範囲で使用するようにしてください。

上記のenumeratorメソッドを使用した例は、Swift 4以降での書き方です。Swift 3以前では、enumeratorメソッドは使用されておらず、代わりにenumerator(at:includingPropertiesForKeys:options:errorHandler:)メソッドが使用されていました。

Swift3

以下は、Swift 3でenumeratorメソッドを使用して、指定されたディレクトリ以下のすべてのファイルとディレクトリを取得する例です。

Swift 3でenumeratorメソッドを使用して、指定されたディレクトリ以下のすべてのファイルとディレクトリを取得する例
if let documentDirectory = NSSearchPathForDirectoriesInDomains(.documentDirectory, .userDomainMask, true).first {
    let fileManager = FileManager.default
    let keys = [URLResourceKey.isDirectoryKey, URLResourceKey.localizedNameKey]
    if let enumerator = fileManager.enumerator(atPath: documentDirectory) {
        while let url = enumerator.nextObject() as? URL {
            do {
                let resourceValues = try url.resourceValues(forKeys: Set(keys))
                if resourceValues.isDirectory == true {
                    print("\(url.path) is a directory")
                } else {
                    print("\(url.path) is a file")
                }
            } catch {
                print("Error: \(error.localizedDescription)")
            }
        }
    }
}

上記のコードでは、enumeratorメソッドを使用する代わりに、enumerator(at:includingPropertiesForKeys:options:errorHandler:)メソッドを使用しています。このメソッドも、深い列挙を行うことができます。enumeratorメソッドと同様に、URLResourceKeyを使用して、ディレクトリであるかどうかを判断しています。

ただし、Swift 3以前でenumeratorメソッドを使用することはできませんので、ご注意ください。

Swift 2, 1でファイルの一覧を取得する場合は、NSFileManagerクラスのcontentsOfDirectoryAtPathメソッドを使用します。

以下は、Swift 2での例です。

Swift2

Swift 2でファイルの一覧を取得する場合は、NSFileManagerクラスのcontentsOfDirectoryAtPathメソッドを使用例
let fileManager = NSFileManager.defaultManager()
if let documentDirectory = NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true).first {
    let filePaths = try? fileManager.contentsOfDirectoryAtPath(documentDirectory)
    if let paths = filePaths {
        for path in paths {
            print(path)
        }
    }
}

 上記のコードでは、NSFileManagerクラスのdefaultManagerメソッドを使用してファイルマネージャのインスタンスを取得し、contentsOfDirectoryAtPathメソッドで指定されたディレクトリのファイル一覧を取得しています。取得したファイルパスは、for-inループを使用して順番に出力されます。

Swift1

Swift 1では、NSSearchPathForDirectoriesInDomainsメソッドの引数の指定が異なるため、以下のようなコードになります。

Swift 1でファイルの一覧を表示する例
let fileManager = NSFileManager.defaultManager()
if let documentDirectory = NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true).first as String? {
    let filePaths = fileManager.contentsOfDirectoryAtPath(documentDirectory)
    if let paths = filePaths {
        for path in paths {
            print(path)
        }
    }
}

 Swift 1やSwift 2では、NSFileManagerクラスのcontentsOfDirectoryAtPathメソッドを使用することで、ファイルの一覧を取得することができます。ただし、Swift 3以降で推奨されるenumeratorメソッドを使用することで、より柔軟なファイル操作が可能になりますので、可能であればenumeratorメソッドを使用することをお勧めします。

ObjectiveC についても同じようにファイル一覧を出す方法

 Objective-Cでファイルの一覧を取得する場合は、NSFileManagerクラスのcontentsOfDirectoryAtPath:error:メソッドを使用します。

以下は、Objective-Cでの例です。

Objective-Cでファイルの一覧を表示する例
NSFileManager *fileManager = [NSFileManager defaultManager];
NSString *documentDirectory = [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) firstObject];
NSError *error;
NSArray *filePaths = [fileManager contentsOfDirectoryAtPath:documentDirectory error:&error];
if (filePaths) {
    for (NSString *filePath in filePaths) {
        NSLog(@"%@", filePath);
    }
} else {
    NSLog(@"Error: %@", error);
}

上記のコードでは、NSFileManagerクラスのdefaultManagerメソッドを使用してファイルマネージャのインスタンスを取得し、contentsOfDirectoryAtPath:error:メソッドで指定されたディレクトリのファイル一覧を取得しています。取得したファイルパスは、for-inループを使用して順番にログに出力されます。

ファイルの一覧を取得するときには、エラーの発生を考慮する必要があります。contentsOfDirectoryAtPath:error:メソッドはエラーが発生した場合にはnilを返します。上記の例では、NSErrorオブジェクトを使用してエラーを取得しています。エラーが発生した場合は、必要に応じてエラーをログに出力したり、エラーハンドリングを行うようにしてください。

上記の例では、NSErrorオブジェクトを使用してエラーを取得しています。NSErrorオブジェクトがnilでない場合には、エラーの内容をログに出力しています。

NSErrorオブジェクトには、エラーが発生した原因に関する情報が含まれています。NSErrorオブジェクトのdomainプロパティとcodeプロパティを使用することで、エラーの原因を特定することができます。詳細については、公式ドキュメントを参照してください。

まとめ

割と長くなりましたが、Swiftのバージョンによって使われていないものもあるので、バージョンによる書き方をアップデートしてみてください。またSwiftだけじゃなくて現場によってはObjective-Cで書かれている現場もあったりするので焦らず調べてみてください。


関連記事


制作チーム:サンストライプ

sunstripe_logo.png
http://sunstripe.main.jp/
サンストライプについて
制作チームのサンストライプのコンテンツについては以下の通り

月3回のコンテンツをリリースしています。
参加者は、プログラマーやデザイナー、イラストレーター、声優やVTuberなどが募集中です。

関連プロジェクト・支援団体

その他協力応援 / 支援者の集い

0
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
0
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?