Как мне рекурсивно получить все пути к ресурсам в моем комплекте в iOS? - PullRequest
16 голосов
/ 29 апреля 2011

В моем комплекте приложений есть «разархивированная» папка. Мне нужно получить пути к ресурсам для всех файлов типа TXT. Я использовал это,

NSArray *filePaths = [NSBundle pathsForResourcesOfType:@"txt" inDirectory:[[[NSBundle mainBundle] bundlePath] stringByAppendingString:@"/unzipped"]];

Но он получает файлы только в первом каталоге. Есть ли где-то рекурсивная версия этого, которую мне просто не хватает?

Ответы [ 5 ]

20 голосов
/ 02 мая 2011

Я получил это с помощью кода, отправленного @rekle в качестве отправной точки. Хитрость заключается в том, чтобы использовать NSDirectoryEnumerator, который будет делать это рекурсивно. Вот функция, которую я написал на случай, если кому-то понадобится.

- (NSArray *)recursivePathsForResourcesOfType:(NSString *)type inDirectory:(NSString *)directoryPath{

    NSMutableArray *filePaths = [[NSMutableArray alloc] init];

    // Enumerators are recursive
    NSDirectoryEnumerator *enumerator = [[[NSFileManager defaultManager] enumeratorAtPath:directoryPath] retain];

    NSString *filePath;

    while ((filePath = [enumerator nextObject]) != nil){

        // If we have the right type of file, add it to the list
        // Make sure to prepend the directory path
        if([[filePath pathExtension] isEqualToString:type]){
            [filePaths addObject:[directoryPath stringByAppendingPathComponent:filePath]];
        }
    }

    [enumerator release];

    return [filePaths autorelease];
}

Swift, используя NSURL

func recursivePathsForResources(type type: String) -> [NSURL] {

    // Enumerators are recursive
    let enumerator = NSFileManager.defaultManager().enumeratorAtPath(bundlePath)
    var filePaths = [NSURL]()

    while let filePath = enumerator?.nextObject() as? String {

        if NSURL(fileURLWithPath: filePath).pathExtension == type {
            filePaths.append(bundleURL.URLByAppendingPathComponent(filePath))
        }
    }

    return filePaths
}
8 голосов
/ 04 февраля 2014

Просто обновление этого метода, чтобы разрешить любые типы файлов (ноль) и предоставить правильный формат пути

- (NSArray *)recursivePathsForResourcesOfType:(NSString *)type inDirectory:(NSString *)directoryPath{

    NSMutableArray *filePaths = [[NSMutableArray alloc] init];
    NSDirectoryEnumerator *enumerator = [[NSFileManager defaultManager] enumeratorAtPath:directoryPath];

    NSString *filePath;

    while ((filePath = [enumerator nextObject]) != nil){
        if (!type || [[filePath pathExtension] isEqualToString:type]){
            [filePaths addObject:[directoryPath stringByAppendingPathComponent:filePath]];
        }
    }

    return filePaths;
}
4 голосов
/ 30 апреля 2011

Попробуйте что-то вроде этого:

NSString *bundlePath = [[NSBundle mainBundle] bundlePath];
NSError *error = nil;
NSArray *files = [[NSFileManager defaultManager] contentsOfDirectoryAtPath:bundlePath error:&error];

'files' будет массивом строк, содержащих все имена файлов в этом пути ...

1 голос
/ 31 мая 2017

Swift 3.1 версия ответа от курилка

func recursivePathsForResources(type: String, in directoryPath: String) -> [String] {
    // Enumerators are recursive
    let enumerator = FileManager.default.enumerator(atPath: directoryPath)
    var filePaths: [String] = []

    while let filePath = enumerator?.nextObject() as? String {

        if URL(fileURLWithPath: filePath).pathExtension == type {
            filePaths.append(directoryPath.byAppending(pathComponent: filePath))
        }
    }
    return filePaths
}
0 голосов
/ 06 ноября 2012

Перед копированием подумайте о создании символической ссылки. Вы можете создать только одну символическую ссылку на папку с большим количеством элементов. И это будет работать в большинстве ситуаций.

Вы будете свободны от создания другого потока, ожидания завершения операции и т. Д.

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