我在这里遇到了一个问题。我正在开发一个读取文件并在 UITableView 中显示其内容的应用程序。我最近意识到文件可能会变得非常大,并且我需要对文件的实际读取进行异步编码。我的项目已经很大了,今天我设法将我已经拥有的代码包装在一个 NSOperation 中。
我所做的是现在在 NSOperation 中调用我的解析器(打开和读取我的文件)。就像这样:
@implementation ReadPcapOperation
@synthesize parser =_parser;
- (id) initWithURL:(NSURL *)url linkedTo:(PacketFlowViewController *)packetController
{
self = [super init];
if (self) {
_parser = [[PcapParser alloc] initWithURL:url linkedTo:packetController];
}
return self;
}
- (void)main {
// a lengthy operation
@autoreleasepool {
if (self.isCancelled)
return;
[_parser read];
}
}
@end
我在这里只给你实现,.h 文件中没有什么重要的。这个 NSOperation 子类在 NSOperation 中被调用:
_queue = [NSOperationQueue new];
_queue.name = @"File Parsing Queue";
_queue.maxConcurrentOperationCount = 1;
[_queue addOperation:_readFileOperation];
_readFileOperation 是上述 ReadPcapOperation 的一个实例。
现在,当我测试我的代码时,仍然没有区别,当我打开文件时 UI 仍然被阻止,而文件内容正在加载到我的 UITableView 中。我在这种情况下进行了测试:
[NSThread isMainThread]
这个测试从 ReadPcapOperation 返回NO in main,这很好,正是我需要的。但是当我把它放在方法“read”中时,这个测试返回YES ,消息从 ReadPcapOperation 发送到 main 内的对象!所以我的整个代码仍在主线程上运行并阻塞了我的 UI。
伙计们,我在这里错过了什么?
如果您需要更多解释,请告诉我!
编辑 :
奇怪的是:我将发布应该在后台线程中执行的代码的一部分。
- (void) read
{
if ([NSThread isMainThread])
NSLog(@"read: IT S MAIN THREAD");
else
NSLog(@"read: IT S NOT MAIN THREAD");
[_fileStream open];
}
- (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode {
switch(eventCode)
{
case NSStreamEventOpenCompleted:
{
//We read the pcap file header
[self readGlobalHeader];
[self readNextPacket];
break;
}
case NSStreamEventHasBytesAvailable:
{
//We read all packets
[self readNextPacket];
break;
}
case NSStreamEventNone:
{
break;
}
case NSStreamEventHasSpaceAvailable:
{
break;
}
case NSStreamEventEndEncountered:
{
NSLog(@"End encountered !");
[_fileStream close];
[_fileStream removeFromRunLoop:[NSRunLoop currentRunLoop]
forMode:NSDefaultRunLoopMode];
//_fileStream = nil;
break;
}
case NSStreamEventErrorOccurred:
{
NSError *theError = [stream streamError];
NSLog(@"Error %i stream event occured. Domain : %@.", theError.code, theError.domain);
[stream close];
break;
}
}
}
- (void) readGlobalHeader
{
if ([NSThread isMainThread])
NSLog(@"readGlobalHeader: IT S MAIN THREAD");
else
NSLog(@"readGlobalHeader: IT S NOT MAIN THREAD");
int sizeOfGlobalHeader = 24;
在这里你可以看到我直接从 NSOperation 调用的方法 read。那里的日志说:“不是主线程”。到目前为止,一切都很好。读取打开 NSInputStreamObject,然后委托将调用“handleEvent”,此时我读取字节。当我调用“readGlobalHeader”并读取文件的第一个字节时,日志是“MAIN THREAD”。它不应该在后台线程中吗?我真的迷路了!
可能需要注意的是,当我初始化流时,在调用“读取”之前,我用这行代码设置了它(我不确定这是原因):
[_fileStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
编辑 2: 这是断点后的回溯。行为就像我上面描述的那样。(没有足够的声誉来发布图片)