某天,在ios8.1下运行了一个之前没有问题的老项目,登录的时候出现了:NSXMLParser does not support reentrant parsing。
直接翻译成中文,系统的xml解析不支持折返解析,不理解!于是开始写demo测试问题。
因为此项目登录接口将2份用途不同的数据合并在同一个接口返回,代码中对此xml数据解析了2次,所以首先模拟对一份数据进行多次解析的情况
1、demo1,在同一个函数中,调用2次xml解析代码
NSString *xmlStr = @"<test><goods>asdf</goods><goods>asdf</goods></test>"; NSData *xmlData = [xmlStr dataUsingEncoding:NSUTF8StringEncoding allowLossyConversion:YES]; NSXMLParser * parser = [[NSXMLParser alloc] initWithData:xmlData]; parser.delegate = self; [parser parse]; NSXMLParser * parser1 = [[NSXMLParser alloc] initWithData:xmlData]; parser1.delegate = self; [parser1 parse];
run后发现运行正常,并未出现报错。再研究下老项目的代码,发现老项目中首先对xml数据是否登录成功进行了一次解析,并在解析成功后调用的block中,再进行了另一次解析获取另一份数据,那么模拟之
2、demo2,在xml解析的delegate回调中,再进行xml解析
- (void)btnPublishOnClick { NSString *xmlStr = @"<test><goods>asdf</goods><goods>asdf</goods></test>"; NSData *xmlData = [xmlStr dataUsingEncoding:NSUTF8StringEncoding allowLossyConversion:YES]; NSXMLParser * parser = [[NSXMLParser alloc] initWithData:xmlData]; parser.delegate = self; self.curParserType = @"first"; [parser parse]; return; } - (void)parserDidEndDocument:(NSXMLParser *)parser { NSString *xmlStr = @"<test><goods>asdf</goods><goods>asdf</goods></test>"; NSData *xmlData = [xmlStr dataUsingEncoding:NSUTF8StringEncoding allowLossyConversion:YES]; if ([self.curParserType isEqualToString:@"first"]) { NSXMLParser * parser1 = [[NSXMLParser alloc] initWithData:xmlData]; self.curParserType = @"second"; parser1.delegate = self; [parser1 parse]; } }
果然,运行后,出现NSXMLParser does not support reentrant parsing的报错,那么可以得出以下结论:
在xml解析的delegate回调中(如:parserDidEndDocument),进行二次xml解析,就会出现NSXMLParser does not support reentrant parsing的报错
解决办法:
1、开线程,将二次解析xml的代码放在线程中执行
- (void)parserDidEndDocument:(NSXMLParser *)parser { NSString *xmlStr = @"<test><goods>asdf</goods><goods>asdf</goods></test>"; NSData *xmlData = [xmlStr dataUsingEncoding:NSUTF8StringEncoding allowLossyConversion:YES]; if ([self.curParserType isEqualToString:@"first"]) { dispatch_queue_t reentrantAvoidanceQueue = dispatch_queue_create("reentrantAvoidanceQueue", DISPATCH_QUEUE_SERIAL); dispatch_async(reentrantAvoidanceQueue, ^{ NSXMLParser * parser1 = [[NSXMLParser alloc] initWithData:xmlData]; self.curParserType = @"second"; parser1.delegate = self; [parser1 parse]; }); dispatch_sync(reentrantAvoidanceQueue, ^{ }); } }
2、使用performSelector,将二次解析xml的代码延迟执行
- (void)sencondParser:(NSData *)xmlData { NSXMLParser * parser1 = [[NSXMLParser alloc] initWithData:xmlData]; self.curParserType = @"second"; parser1.delegate = self; [parser1 parse]; } - (void)parserDidEndDocument:(NSXMLParser *)parser { NSString *xmlStr = @"<test><goods>asdf</goods><goods>asdf</goods></test>"; NSData *xmlData = [xmlStr dataUsingEncoding:NSUTF8StringEncoding allowLossyConversion:YES]; if ([self.curParserType isEqualToString:@"first"]) { [self performSelector:@selector(sencondParser:) withObject:xmlData afterDelay:0.1]; } }
时间: 2024-12-04 22:58:59