web-dev-qa-db-ja.com

Objective-C:行ごとにファイルを読み取る

Objective-Cで大きなテキストファイルを処理する適切な方法は何ですか?各行を個別に読み取り、各行をNSStringとして処理する必要があるとします。これを行う最も効率的な方法は何ですか?

1つの解決策は、NSStringメソッドを使用することです。

+ (id)stringWithContentsOfFile:(NSString *)path 
      encoding:(NSStringEncoding)enc 
      error:(NSError **)error 

次に、改行セパレーターで行を分割し、配列内の要素を反復処理します。ただし、これはかなり効率が悪いようです。ファイルを一度にすべて読み込むのではなく、ファイルをストリームとして扱い、各行を列挙する簡単な方法はありませんか? JavaのJava.io.BufferedReaderに似ています。

136
Jonathan

それは素晴らしい質問です。 @ Diederikは良い答えを持っていると思いますが、Cocoaがあなたがやりたいことを正確に行うメカニズムを持っていないのは残念です。

NSInputStream は、Nバイトのチャンクを読み取ることができます(Java.io.BufferedReaderに非常に似ています)が、それを自分でNSStringに変換してから、改行(またはその他の区切り文字)をスキャンする必要があります次の読み取りのために残りの文字を保存するか、改行がまだ読み取られていない場合はさらに文字を読み取ります。 ( NSFileHandle を使用すると、NSDataを読み取って、NSStringに変換できますが、基本的には同じプロセスです。)

Appleには Stream Programming Guide があり、詳細を記入するのに役立ちます。また、 this SO question も対処する場合に役立ちますuint8_t*バッファあり。

このような文字列を頻繁に(特にプログラムの異なる部分で)読み取る場合は、この動作を詳細を処理できるクラスにカプセル化するか、NSInputStreamをサブクラス化することをお勧めします(それは- サブクラス化するように設計されている )、必要なものを正確に読み取ることができるメソッドを追加します。

記録のために、これは追加する素晴らしい機能であると思います、そして、私はこれを可能にする何かのために拡張要求を提出します。 :-)


Edit:このリクエストは既に存在することが判明します。このために2006年以降のレーダーがあります(Apple内部の人々にはrdar:// 4742914)。

63
Quinn Taylor

これは、StringからTextを一般的に読み取る場合に機能します。より長いテキスト(テキストの大きなサイズ)を読みたい場合は、バッファリングされた(テキストのサイズをメモリ領域に確保する)など、ここで他の人が言及した方法を使用してください

テキストファイルを読んだとしましょう。

NSString* filePath = @""//file path...
NSString* fileRoot = [[NSBundle mainBundle] 
               pathForResource:filePath ofType:@"txt"];

新しい行を取り除きたい。

// read everything from text
NSString* fileContents = 
      [NSString stringWithContentsOfFile:fileRoot 
       encoding:NSUTF8StringEncoding error:nil];

// first, separate by new line
NSArray* allLinedStrings = 
      [fileContents componentsSeparatedByCharactersInSet:
      [NSCharacterSet newlineCharacterSet]];

// then break down even further 
NSString* strsInOneLine = 
      [allLinedStrings objectAtIndex:0];

// choose whatever input identity you have decided. in this case ;
NSArray* singleStrs = 
      [currentPointString componentsSeparatedByCharactersInSet:
      [NSCharacterSet characterSetWithCharactersInString:@";"]];

そこにあります。

94
Yoon Lee

これでうまくいくはずです:

#include <stdio.h>

NSString *readLineAsNSString(FILE *file)
{
    char buffer[4096];

    // tune this capacity to your liking -- larger buffer sizes will be faster, but
    // use more memory
    NSMutableString *result = [NSMutableString stringWithCapacity:256];

    // Read up to 4095 non-newline characters, then read and discard the newline
    int charsRead;
    do
    {
        if(fscanf(file, "%4095[^\n]%n%*c", buffer, &charsRead) == 1)
            [result appendFormat:@"%s", buffer];
        else
            break;
    } while(charsRead == 4095);

    return result;
}

次のように使用します。

FILE *file = fopen("myfile", "r");
// check for NULL
while(!feof(file))
{
    NSString *line = readLineAsNSString(file);
    // do stuff with line; line is autoreleased, so you should NOT release it (unless you also retain it beforehand)
}
fclose(file);

このコードは、一度に最大4095まで、ファイルから非改行文字を読み取ります。 4095文字より長い行がある場合、改行またはファイルの終わりに達するまで読み取りを続けます。

:このコードはテストしていません。使用する前にテストしてください。

34
Adam Rosenfield

Mac OS XはUnix、Objective-CはCスーパーセットです。そのため、<stdio.h>の旧式のfopenおよびfgetsを使用できます。動作することが保証されています。

[NSString stringWithUTF8String:buf]は、C文字列をNSStringに変換します。他のエンコーディングで文字列を作成し、コピーせずに作成する方法もあります。

12
Kornel

ファイルストリームの基本的な実装を持つNSInputStreamを使用できます。バイトをバッファに読み込むことができます(read:maxLength:メソッド)。バッファーの改行を自分でスキャンする必要があります。

9
diederikh

Cocoa/Objective-Cでテキストファイルを読み取る適切な方法は、AppleのStringプログラミングガイドに記載されています。 ファイルの読み取りと書き込み のセクションは、まさにあなたが求めているものでなければなりません。 PS:「ライン」とは何ですか? 「\ n」で区切られた文字列の2つのセクション?または「\ r」?または「\ r\n」?それとも、実際には段落の後ですか?前述のガイドには、文字列を行または段落に分割するセクションも含まれています。 (このセクションは「段落と改行」と呼ばれ、上記で指摘したページの左側のメニューにリンクされています。残念ながら、このサイトでは、複数のURLを投稿することはできません。まだ信頼できるユーザーではありません。)

Knuthを言い換えると、早すぎる最適化はすべての悪の根源です。 「ファイル全体をメモリに読み込む」のが遅いと単純に仮定しないでください。ベンチマークしましたか? 実際にファイル全体をメモリに読み込むことを知っていますか?たぶん、それは単にプロキシオブジェクトを返し、あなたが文字列を消費するときに舞台裏で読み続けますか? (免責事項:NSStringが実際にこれを行うかどうかはわかりません。おそらく可能です。)ポイントは、最初に文書化された方法を実行することです。次に、ベンチマークで、希望するパフォーマンスが得られないことがわかった場合は、最適化してください。

6
Stig Brautaset

これらの回答の多くは長いコードの塊であるか、ファイル全体を読み取ります。私はこのタスクにcメソッドを使用するのが好きです。

FILE* file = fopen("path to my file", "r");

size_t length;
char *cLine = fgetln(file,&length);

while (length>0) {
    char str[length+1];
    strncpy(str, cLine, length);
    str[length] = '\0';

    NSString *line = [NSString stringWithFormat:@"%s",str];        
    % Do what you want here.

    cLine = fgetln(file,&length);
}

Fgetlnは改行文字を保持しないことに注意してください。また、NULL終端用のスペースを作りたいので、strの長さを+1します。

5
DCurro

他の人が答えたように、NSInputStreamとNSFileHandleはどちらも素晴らしいオプションですが、NSDataとメモリマッピングを使用してかなりコンパクトに行うこともできます。

BRLineReader.h

#import <Foundation/Foundation.h>

@interface BRLineReader : NSObject

@property (readonly, nonatomic) NSData *data;
@property (readonly, nonatomic) NSUInteger linesRead;
@property (strong, nonatomic) NSCharacterSet *lineTrimCharacters;
@property (readonly, nonatomic) NSStringEncoding stringEncoding;

- (instancetype)initWithFile:(NSString *)filePath encoding:(NSStringEncoding)encoding;
- (instancetype)initWithData:(NSData *)data encoding:(NSStringEncoding)encoding;
- (NSString *)readLine;
- (NSString *)readTrimmedLine;
- (void)setLineSearchPosition:(NSUInteger)position;

@end

BRLineReader.m

#import "BRLineReader.h"

static unsigned char const BRLineReaderDelimiter = '\n';

@implementation BRLineReader
{
    NSRange _lastRange;
}

- (instancetype)initWithFile:(NSString *)filePath encoding:(NSStringEncoding)encoding
{
    self = [super init];
    if (self) {
        NSError *error = nil;
        _data = [NSData dataWithContentsOfFile:filePath options:NSDataReadingMappedAlways error:&error];
        if (!_data) {
            NSLog(@"%@", [error localizedDescription]);
        }
        _stringEncoding = encoding;
        _lineTrimCharacters = [NSCharacterSet whitespaceAndNewlineCharacterSet];
    }

    return self;
}

- (instancetype)initWithData:(NSData *)data encoding:(NSStringEncoding)encoding
{
    self = [super init];
    if (self) {
        _data = data;
        _stringEncoding = encoding;
        _lineTrimCharacters = [NSCharacterSet whitespaceAndNewlineCharacterSet];
    }

    return self;
}

- (NSString *)readLine
{
    NSUInteger dataLength = [_data length];
    NSUInteger beginPos = _lastRange.location + _lastRange.length;
    NSUInteger endPos = 0;
    if (beginPos == dataLength) {
        // End of file
        return nil;
    }

    unsigned char *buffer = (unsigned char *)[_data bytes];
    for (NSUInteger i = beginPos; i < dataLength; i++) {
        endPos = i;
        if (buffer[i] == BRLineReaderDelimiter) break;
    }

    // End of line found
    _lastRange = NSMakeRange(beginPos, endPos - beginPos + 1);
    NSData *lineData = [_data subdataWithRange:_lastRange];
    NSString *line = [[NSString alloc] initWithData:lineData encoding:_stringEncoding];
    _linesRead++;

    return line;
}

- (NSString *)readTrimmedLine
{
    return [[self readLine] stringByTrimmingCharactersInSet:_lineTrimCharacters];
}

- (void)setLineSearchPosition:(NSUInteger)position
{
    _lastRange = NSMakeRange(position, 0);
    _linesRead = 0;
}

@end
3

@porneLが言ったように、C APIは非常に便利です。

NSString* fileRoot = [[NSBundle mainBundle] pathForResource:@"record" ofType:@"txt"];
FILE *file = fopen([fileRoot UTF8String], "r");
char buffer[256];
while (fgets(buffer, 256, file) != NULL){
    NSString* result = [NSString stringWithUTF8String:buffer];
    NSLog(@"%@",result);
}
3
wdanxna

ファイルを1行ずつ読み込む(非常に大きなファイルの場合も)には、次の関数を使用します。

DDFileReader * reader = [[DDFileReader alloc] initWithFilePath:pathToMyFile];
NSString * line = nil;
while ((line = [reader readLine])) {
  NSLog(@"read line: %@", line);
}
[reader release];

または:

DDFileReader * reader = [[DDFileReader alloc] initWithFilePath:pathToMyFile];
[reader enumerateLinesUsingBlock:^(NSString * line, BOOL * stop) {
  NSLog(@"read line: %@", line);
}];
[reader release];

これを可能にするクラスDDFileReaderは次のとおりです。

インターフェイスファイル(.h):

@interface DDFileReader : NSObject {
    NSString * filePath;

    NSFileHandle * fileHandle;
    unsigned long long currentOffset;
    unsigned long long totalFileLength;

    NSString * lineDelimiter;
    NSUInteger chunkSize;
}

@property (nonatomic, copy) NSString * lineDelimiter;
@property (nonatomic) NSUInteger chunkSize;

- (id) initWithFilePath:(NSString *)aPath;

- (NSString *) readLine;
- (NSString *) readTrimmedLine;

#if NS_BLOCKS_AVAILABLE
- (void) enumerateLinesUsingBlock:(void(^)(NSString*, BOOL *))block;
#endif

@end

実装(.m)

#import "DDFileReader.h"

@interface NSData (DDAdditions)

- (NSRange) rangeOfData_dd:(NSData *)dataToFind;

@end

@implementation NSData (DDAdditions)

- (NSRange) rangeOfData_dd:(NSData *)dataToFind {

    const void * bytes = [self bytes];
    NSUInteger length = [self length];

    const void * searchBytes = [dataToFind bytes];
    NSUInteger searchLength = [dataToFind length];
    NSUInteger searchIndex = 0;

    NSRange foundRange = {NSNotFound, searchLength};
    for (NSUInteger index = 0; index < length; index++) {
        if (((char *)bytes)[index] == ((char *)searchBytes)[searchIndex]) {
            //the current character matches
            if (foundRange.location == NSNotFound) {
                foundRange.location = index;
            }
            searchIndex++;
            if (searchIndex >= searchLength) { return foundRange; }
        } else {
            searchIndex = 0;
            foundRange.location = NSNotFound;
        }
    }
    return foundRange;
}

@end

@implementation DDFileReader
@synthesize lineDelimiter, chunkSize;

- (id) initWithFilePath:(NSString *)aPath {
    if (self = [super init]) {
        fileHandle = [NSFileHandle fileHandleForReadingAtPath:aPath];
        if (fileHandle == nil) {
            [self release]; return nil;
        }

        lineDelimiter = [[NSString alloc] initWithString:@"\n"];
        [fileHandle retain];
        filePath = [aPath retain];
        currentOffset = 0ULL;
        chunkSize = 10;
        [fileHandle seekToEndOfFile];
        totalFileLength = [fileHandle offsetInFile];
        //we don't need to seek back, since readLine will do that.
    }
    return self;
}

- (void) dealloc {
    [fileHandle closeFile];
    [fileHandle release], fileHandle = nil;
    [filePath release], filePath = nil;
    [lineDelimiter release], lineDelimiter = nil;
    currentOffset = 0ULL;
    [super dealloc];
}

- (NSString *) readLine {
    if (currentOffset >= totalFileLength) { return nil; }

    NSData * newLineData = [lineDelimiter dataUsingEncoding:NSUTF8StringEncoding];
    [fileHandle seekToFileOffset:currentOffset];
    NSMutableData * currentData = [[NSMutableData alloc] init];
    BOOL shouldReadMore = YES;

    NSAutoreleasePool * readPool = [[NSAutoreleasePool alloc] init];
    while (shouldReadMore) {
        if (currentOffset >= totalFileLength) { break; }
        NSData * chunk = [fileHandle readDataOfLength:chunkSize];
        NSRange newLineRange = [chunk rangeOfData_dd:newLineData];
        if (newLineRange.location != NSNotFound) {

            //include the length so we can include the delimiter in the string
            chunk = [chunk subdataWithRange:NSMakeRange(0, newLineRange.location+[newLineData length])];
            shouldReadMore = NO;
        }
        [currentData appendData:chunk];
        currentOffset += [chunk length];
    }
    [readPool release];

    NSString * line = [[NSString alloc] initWithData:currentData encoding:NSUTF8StringEncoding];
    [currentData release];
    return [line autorelease];
}

- (NSString *) readTrimmedLine {
    return [[self readLine] stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
}

#if NS_BLOCKS_AVAILABLE
- (void) enumerateLinesUsingBlock:(void(^)(NSString*, BOOL*))block {
  NSString * line = nil;
  BOOL stop = NO;
  while (stop == NO && (line = [self readLine])) {
    block(line, &stop);
  }
}
#endif

@end

クラスは Dave DeLong によって行われました

3
lukaswelte

この答えはObjCではなくCです。

ObjCは 'C'ベースなので、fgetsを使用しないのはなぜですか?

そして、はい、ObjCには独自のメソッドがあると確信しています-私はそれが何であるかを知るにはまだ十分に熟練していません:)

0
KevinDTimm

私が試した他のすべての答えが何らかの形で不足したため、これを追加しています。次のメソッドは、大きなファイル、任意の長い行、および空の行を処理できます。実際のコンテンツでテストされており、出力から改行文字を取り除きます。

- (NSString*)readLineFromFile:(FILE *)file
{
    char buffer[4096];
    NSMutableString *result = [NSMutableString stringWithCapacity:1000];

    int charsRead;
    do {
        if(fscanf(file, "%4095[^\r\n]%n%*[\n\r]", buffer, &charsRead) == 1) {
            [result appendFormat:@"%s", buffer];
        }
        else {
            break;
        }
    } while(charsRead == 4095);

    return result.length ? result : nil;
}

クレジットは@Adam Rosenfieldと@sooopに送られます

0
Blago

@lukaswelteによる応答と Dave DeLong のコードが非常に役立つことがわかりました。この問題の解決策を探していましたが、\r\nだけでなく\nで大きなファイルを解析する必要がありました。

記述されたコードには、複数の文字で解析する場合のバグが含まれています。以下のようにコードを変更しました。

.hファイル:

#import <Foundation/Foundation.h>

@interface FileChunkReader : NSObject {
    NSString * filePath;

    NSFileHandle * fileHandle;
    unsigned long long currentOffset;
    unsigned long long totalFileLength;

    NSString * lineDelimiter;
    NSUInteger chunkSize;
}

@property (nonatomic, copy) NSString * lineDelimiter;
@property (nonatomic) NSUInteger chunkSize;

- (id) initWithFilePath:(NSString *)aPath;

- (NSString *) readLine;
- (NSString *) readTrimmedLine;

#if NS_BLOCKS_AVAILABLE
- (void) enumerateLinesUsingBlock:(void(^)(NSString*, BOOL *))block;
#endif

@end

.mファイル:

#import "FileChunkReader.h"

@interface NSData (DDAdditions)

- (NSRange) rangeOfData_dd:(NSData *)dataToFind;

@end

@implementation NSData (DDAdditions)

- (NSRange) rangeOfData_dd:(NSData *)dataToFind {

    const void * bytes = [self bytes];
    NSUInteger length = [self length];

    const void * searchBytes = [dataToFind bytes];
    NSUInteger searchLength = [dataToFind length];
    NSUInteger searchIndex = 0;

    NSRange foundRange = {NSNotFound, searchLength};
    for (NSUInteger index = 0; index < length; index++) {
        if (((char *)bytes)[index] == ((char *)searchBytes)[searchIndex]) {
            //the current character matches
            if (foundRange.location == NSNotFound) {
                foundRange.location = index;
            }
            searchIndex++;
            if (searchIndex >= searchLength)
            {
                return foundRange;
            }
        } else {
            searchIndex = 0;
            foundRange.location = NSNotFound;
        }
    }

    if (foundRange.location != NSNotFound
        && length < foundRange.location + foundRange.length )
    {
        // if the dataToFind is partially found at the end of [self bytes],
        // then the loop above would end, and indicate the dataToFind is found
        // when it only partially was.
        foundRange.location = NSNotFound;
    }

    return foundRange;
}

@end

@implementation FileChunkReader

@synthesize lineDelimiter, chunkSize;

- (id) initWithFilePath:(NSString *)aPath {
    if (self = [super init]) {
        fileHandle = [NSFileHandle fileHandleForReadingAtPath:aPath];
        if (fileHandle == nil) {
            return nil;
        }

        lineDelimiter = @"\n";
        currentOffset = 0ULL; // ???
        chunkSize = 128;
        [fileHandle seekToEndOfFile];
        totalFileLength = [fileHandle offsetInFile];
        //we don't need to seek back, since readLine will do that.
    }
    return self;
}

- (void) dealloc {
    [fileHandle closeFile];
    currentOffset = 0ULL;

}

- (NSString *) readLine {
    if (currentOffset >= totalFileLength)
    {
        return nil;
    }

    @autoreleasepool {

        NSData * newLineData = [lineDelimiter dataUsingEncoding:NSUTF8StringEncoding];
        [fileHandle seekToFileOffset:currentOffset];
        unsigned long long originalOffset = currentOffset;
        NSMutableData *currentData = [[NSMutableData alloc] init];
        NSData *currentLine = [[NSData alloc] init];
        BOOL shouldReadMore = YES;


        while (shouldReadMore) {
            if (currentOffset >= totalFileLength)
            {
                break;
            }

            NSData * chunk = [fileHandle readDataOfLength:chunkSize];
            [currentData appendData:chunk];

            NSRange newLineRange = [currentData rangeOfData_dd:newLineData];

            if (newLineRange.location != NSNotFound) {

                currentOffset = originalOffset + newLineRange.location + newLineData.length;
                currentLine = [currentData subdataWithRange:NSMakeRange(0, newLineRange.location)];

                shouldReadMore = NO;
            }else{
                currentOffset += [chunk length];
            }
        }

        if (currentLine.length == 0 && currentData.length > 0)
        {
            currentLine = currentData;
        }

        return [[NSString alloc] initWithData:currentLine encoding:NSUTF8StringEncoding];
    }
}

- (NSString *) readTrimmedLine {
    return [[self readLine] stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
}

#if NS_BLOCKS_AVAILABLE
- (void) enumerateLinesUsingBlock:(void(^)(NSString*, BOOL*))block {
    NSString * line = nil;
    BOOL stop = NO;
    while (stop == NO && (line = [self readLine])) {
        block(line, &stop);
    }
}
#endif

@end
0
hovey

@Adam Rosenfieldの答えから、fscanfのフォーマット文字列は次のように変更されます:

"%4095[^\r\n]%n%*[\n\r]"

osx、linux、windowsの行末で動作します。

0
sooop

カテゴリまたは拡張機能を使用して、生活を少し楽にします。

extension String {

    func lines() -> [String] {
        var lines = [String]()
        self.enumerateLines { (line, stop) -> () in
            lines.append(line)
        }
        return lines
    }

}

// then
for line in string.lines() {
    // do the right thing
}
0
Kaz Yoshikawa