J'essaie de faire un flux d'entrée personnalisé basé sur celui de Dave DeLong here qui permet également de lire des données à partir d'un serveur via NSURL. J'ai jusqu'à présent cette approche, qui fonctionne très bien pour les fichiers locaux:NSFileHandle fileHandleForReadingFromURL quelqu'un peut-il m'expliquer s'il vous plaît?
@interface RJRStreamReader : NSObject {
@private
NSFileHandle *fileHandle;
NSStringEncoding encoding;
NSString *lineDelimiter;
unsigned long long currentOffset;
unsigned long long totalFileLength;
int chunkSize;
}
@property(readwrite, assign) NSStringEncoding encoding;
@property(readwrite, assign) unsigned long long currentOffset;
@property(readwrite, copy) NSString *lineDelimiter;
@property(readwrite, assign) int chunkSize;
@property(readonly) unsigned long long totalFileLength;
-(id) initWithLocalFile:(NSString *) fileName;
-(id) initWithURL:(NSURL *) remoteURL;
-(id) initWithFileHandle:(NSFileHandle *) fh;
-(NSString *) readToEnd;
-(NSString *) readLine;
/**
@summary Reads a block of bytes from a stream
@param blockLen the number of bytes to read
@returns a string containing the data from the bytes read
*/
-(NSString *) readBlock:(int) blockLen;
@end
Et ma mise en œuvre:
@implementation RJRStreamReader
@synthesize currentOffset, lineDelimiter, encoding, chunkSize, totalFileLength;
-(id) initWithLocalFile:(NSString *)fileName
{
if (self = [super init])
{
fileHandle = [NSFileHandle fileHandleForReadingAtPath:fileName];
if (fileHandle == nil)
{
[self release];
return nil;
}
chunkSize = 10;
encoding = NSUTF8StringEncoding;
lineDelimiter = [[NSString alloc] initWithString:@"\n"];
[fileHandle retain];
currentOffset = 0ULL;
[fileHandle seekToEndOfFile];
totalFileLength = [fileHandle offsetInFile];
}
return self;
}
-(id) initWithURL:(NSURL *)remoteURL
{
if (self = [super init])
{
NSError *err = nil;
fileHandle = [NSFileHandle fileHandleForReadingFromURL:remoteURL error:&err];
if (err)
{
NSLog(@"Error occurred, aborting. Details: %@", err);
[self release];
return nil;
}
chunkSize = 10;
encoding = NSUTF8StringEncoding;
lineDelimiter = [[NSString alloc] initWithString:@"\n"];
[fileHandle retain];
currentOffset = 0ULL;
[fileHandle seekToEndOfFile];
totalFileLength = [fileHandle offsetInFile];
}
return self;
}
-(id) initWithFileHandle:(NSFileHandle *) fh
{
if (self = [super init])
{
fileHandle = fh;
if (!fh)
{
[self release];
[NSException raise:@"FH cannot be nil!" format:@"FH cannot be nil!"];
return nil;
}
chunkSize = 10;
encoding = NSUTF8StringEncoding;
lineDelimiter = [[NSString alloc] initWithString:@"\n"];
[fileHandle retain];
currentOffset = 0ULL;
[fileHandle seekToEndOfFile];
totalFileLength = [fileHandle offsetInFile];
}
return self;
}
-(NSString *) readBlock:(int)blockLen
{
if (currentOffset >= totalFileLength) { return nil; }
[fileHandle seekToFileOffset:currentOffset];
NSData *data = [fileHandle readDataOfLength:blockLen];
currentOffset += blockLen;
[fileHandle seekToFileOffset:currentOffset];
return [[[NSString alloc] initWithData:data encoding:encoding] autorelease];
}
-(NSString *) readLine
{
/*
if you want to see the code for this method, visit this link:
https://stackoverflow.com/questions/3707427/how-to-read-data-from-nsfilehandle-line-by-line
it is exactly the same
*/
}
-(NSString *) readToEnd
{
if (currentOffset >= totalFileLength) { return nil; }
[fileHandle seekToFileOffset:currentOffset];
NSData *data = [fileHandle readDataToEndOfFile];
currentOffset = totalFileLength;
[fileHandle seekToEndOfFile];
return [[[NSString alloc] initWithData:data encoding:encoding] autorelease];
}
-(void) dealloc
{
[fileHandle closeFile];
[fileHandle release];
[lineDelimiter release];
[super dealloc];
}
@end
Le problème est, lorsque je tente de l'utiliser comme ceci:
RJRStreamReader *stream = [[RJRStreamReader alloc] initWithURL:[NSURL URLWithString:@"http://www.stackoverflow.com/robots.txt"];
NSString *s = [stream readToEnd];
s
sera toujours vide car le NSFileHandle
retourné par [NSFileHandle fileHandleForReadingFromURL:remoteURL]
semble toujours renvoyer zéro, sans aucune erreur.
Est-ce un bug dans mon code ou une fonctionnalité non documentée?
Merci