两种方法:1.ios中已有的NSXMLParse类@H_502_3@
2.Google的GdataxML也是基于libxml2的,因此在使用GdataxML之前,你需要先导入libxml2。@H_502_3@ @H_404_22@这里详解ios中的nsxmlparse:@H_502_3@
NSXMLParser实现的是sax方法解析xml文件。@H_502_3@
dom实现的原理是把整个xml文档一次性读出,放在一个树型结构里。在需要的时候,查找特定节点,然后对节点进行读或写。他的主要优势是实现简单,读写平衡;缺点是比较占内存,因为他要把整个xml文档都读入内存,文件越大,这种缺点就越明显。@H_502_3@
sax的实现方法和dom不同。他只在xml文档中查找特定条件的内容,并且只提取需要的内容。这样做占用内存小,灵活,正好满足我们的需求。他的缺点就是写,有些资料介绍了写入的方法,但是我感觉这对本例没有必要。@H_502_3@
运行NSXMLParser涉及设置、运行和响应结果。@H_502_3@
1)启动NSXMLParser@H_502_3@
要使用NSXMLParser要先创建它,设置各种属性,主要用到以下几个方法:@H_502_3@
initWithContentsOfURL通过NSURL创建解析器@H_502_3@
initWithData通过NSData创建解析器@H_502_3@
setDelegate为解析器定义委托@H_502_3@
parse运行解析器@H_502_3@
2)充当委托@H_502_3@
//发现元素开始符的处理函数(即报告元素的开始以及元素的属性)@H_502_3@
- (void)parser:(NSXMLParser *)parser@H_502_3@
didStartElement:(NSString *)elementName@H_502_3@
namespaceURI:(NSString *)namespaceURI@H_502_3@
qualifiedName:(NSString *)qName@H_502_3@
attributes:(NSDictionary *)attributeDict@H_502_3@
@H_502_3@
//处理标签包含内容字符(报告元素的所有或部分内容)@H_502_3@
- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string@H_502_3@
@H_502_3@
//发现元素结束符的处理函数,保存元素各项目数据(即报告元素的结束标记)@H_502_3@
- (void)parser:(NSXMLParser *)parser@H_502_3@
didEndElement:(NSString *)elementName@H_502_3@
namespaceURI:(NSString *)namespaceURI@H_502_3@
qualifiedName:(NSString *)qName@H_502_3@
@H_502_3@
//报告解析的结束@H_502_3@
- (void)parserDidEndDocument:(NSXMLParser *)parser@H_502_3@
@H_502_3@
- (void)paser:parserErrorOccured@H_502_3@
1.申明了一个xml处理协议,用到NSXMLParser协议,用来具体处理我们要解析的xml文件,把需要的url取出来,存到指定的data里。@H_502_3@
*/@H_502_3@
//多线程处理类的定义,方法实现,与向主线程回传data@H_502_3@
@classAppRecord;@H_502_3@
@H_502_3@
//@protocol ParSEOperationDelegate;@H_502_3@
@protocolParSEOperationDelegate@H_502_3@
- (void)didFinishParsing:(NSArray*)appList;@H_502_3@
- (void)parseErrorOccurred:(NSError*)error;@H_502_3@
@end@H_502_3@
@H_502_3@
@interfaceParSEOperation : NSOperation <NSXMLParserDelegate> {@H_502_3@
@private@H_502_3@
id<ParSEOperationDelegate>delegate;//,NSXMLParserDelegate @H_502_3@
@H_502_3@
NSData *dataToParse;@H_502_3@
@H_502_3@
NSMutableArray *workingArray;@H_502_3@
AppRecord *workingEntry;@H_502_3@
NSMutableString*workingPropertyString;@H_502_3@
NSArray *elementsToParse;@H_502_3@
BOOL storingCharacterData;@H_502_3@
}@H_502_3@
@H_502_3@
- (id)initWithData:(NSData*)data delegate:(id<ParSEOperationDelegate>)theDelegate;@H_502_3@
@H_502_3@
@end@H_502_3@
- (id)initWithData:(NSData*)data delegate:(id<ParSEOperationDelegate>)theDelegate@H_502_3@
{@H_502_3@
self= [superinit];@H_502_3@
if(self!=nil)@H_502_3@
{@H_502_3@
self.dataToParse= data;@H_502_3@
self.delegate= theDelegate;@H_502_3@
self.elementsToParse= [NSArrayarrayWithObjects:kIDStr,kNameStr,kImageStr,kArtistStr,nil];@H_502_3@
}@H_502_3@
returnself;@H_502_3@
}@H_502_3@
@H_502_3@
#pragma mark -@H_502_3@
#pragma mark RSS processing@H_502_3@
@H_502_3@
- (void)parser:(NSXMLParser*)parser didStartElement:(NSString*)elementName@H_502_3@
namespaceURI:(NSString*)namespaceURI@H_502_3@
qualifiedName:(NSString*)qName@H_502_3@
attributes:(NSDictionary*)attributeDict@H_502_3@
{@H_502_3@
// entry: { id (link),im:name (app name),im:image (variable height) }@H_502_3@
//@H_502_3@
if([elementNameisEqualToString:kEntryStr])@H_502_3@
{@H_502_3@
self.workingEntry= [[[AppRecordalloc]init]autorelease];@H_502_3@
}@H_502_3@
storingCharacterData= [elementsToParsecontainsObject:elementName];@H_502_3@
}@H_502_3@
@H_502_3@
- (void)parser:(NSXMLParser*)parser didEndElement:(NSString*)elementName@H_502_3@
namespaceURI:(NSString*)namespaceURI@H_502_3@
qualifiedName:(NSString*)qName@H_502_3@
{@H_502_3@
if(self.workingEntry)@H_502_3@
{@H_502_3@
if(storingCharacterData)@H_502_3@
{@H_502_3@
NSString*trimmedString = [workingPropertyStringstringByTrimmingCharactersInSet:@H_502_3@
[NSCharacterSetwhitespaceAndNewlineCharacterSet]];@H_502_3@
[workingPropertyStringsetString:@""];// clear the string for next time@H_502_3@
if([elementNameisEqualToString:kIDStr])@H_502_3@
{@H_502_3@
self.workingEntry.appURLString= trimmedString;@H_502_3@
}@H_502_3@
elseif([elementNameisEqualToString:kNameStr])@H_502_3@
{ @H_502_3@
self.workingEntry.appName= trimmedString;@H_502_3@
}@H_502_3@
elseif([elementNameisEqualToString:kImageStr])@H_502_3@
{@H_502_3@
self.workingEntry.imageURLString= trimmedString;@H_502_3@
}@H_502_3@
elseif([elementNameisEqualToString:kArtistStr])@H_502_3@
{@H_502_3@
self.workingEntry.artist= trimmedString;@H_502_3@
}@H_502_3@
}@H_502_3@
elseif([elementNameisEqualToString:kEntryStr])@H_502_3@
{@H_502_3@
[self.workingArrayaddObject:self.workingEntry]; @H_502_3@
self.workingEntry=nil;@H_502_3@
}@H_502_3@
}@H_502_3@
@H_502_3@
}@H_502_3@
@H_502_3@
- (void)parser:(NSXMLParser*)parser foundCharacters:(NSString*)string@H_502_3@
{@H_502_3@
if(storingCharacterData)@H_502_3@
{@H_502_3@
[workingPropertyStringappendString:string];@H_502_3@
}@H_502_3@
}@H_502_3@
@H_502_3@
@H_502_3@
@H_502_3@
@H_502_3@
@H_502_3@
@H_502_3@
@H_502_3@
iOS中XML解析是本文要介绍的内容,在iPhone开发中,XML的解析有很多选择,iOSSDK提供了NSXMLParser和libxml2两个类库,另外还有很多第三方类库可选,例如TBXML、TouchXML、KissXML、TinyXML和GdataxML。问题是应该选择哪一个呢?@H_502_3@
解析XML通常有两种方式,DOM 和 SAX:@H_502_3@
DOM解析XML时,读入整个XML文档并构建一个驻留内存的树结构(节点树),通过遍历树结构可以检索任意XML节点,读取它的属性和值。而且通常情况下,可以借助XPath,直接查询XML节点。@H_502_3@
SAX解析XML,是基于事件通知的模式,一边读取XML文档一边处理,不必等整个文档加载完之后才采取操作,当在读取解析过程中遇到需要处理的对象,会发出通知对其进行处理。@H_502_3@
一般在iOS平台下,比较常用的XML解析类库有如下几种:@H_502_3@
NSXMLParser,http://developer.apple.com/library/ios/#documentation/Cocoa/Reference/Foundation/Classes/NSXMLParser_Class/Reference/Reference.html ,这是一个SAX方式解析XML的类库,默认包含在iOS SDK中,使用也比较简单。@H_502_3@
libxml2,http://xmlsoft.org/,是一套默认包含在iOS SDK中的开源类库,它是基于C语言的API,所以使用起来可能不如NSXML方便。这套类库同时支持DOM和SAX解析,libxml2的SAX解析方式还是非常酷的,因为它可以边读取边解析,尤其是在从网上下载一个很大的XML文件,就可以一边下载一边对已经下载好的内容进行解析,极大的提高解析效率。@H_502_3@
TBXML,http://www.tbxml.co.uk/TBXML/TBXML_Free.html,这是一套轻量级的DOM方式的XML解析类库,有很好的性能和低内存占用,不过它不对XML格式进行校验,不支持XPath,并且只支持解析,不支持对XML进行修改。@H_502_3@
TouchXML,https://github.com/TouchCode/TouchXML,这也是一套DOM方式的XML解析类库,支持XPath,不支持XML的修改。@H_502_3@
KissXML,http://code.google.com/p/kissxml/,这是一套基于TouchXML的XML解析类库,和TouchXML相比,支持了XML的修改。@H_502_3@
TinyXML,http://www.grinninglizard.com/tinyxml/,这是一套小巧的基于C语言的DOM方式进行XML解析的类库,支持对XML的读取和修改,不直接支持XPath,需要借助另一个相关的类库TinyXPath才可以支持XPath。@H_502_3@
GdataxML,http://code.google.com/p/gdata-objectivec-client/source/browse/trunk/Source/XMLSupport/,这是一套Google开发的DOM方式XML解析类库,支持读取和修改XML文档,支持XPath方式查询。@H_502_3@
那么对于如何在项目中选择合适的XML解析类库呢?网上已经有人对这几款XML类库做过分析和对比,可参考《How To Choose The Best XML Parser for Your iPhone Project》http://www.raywenderlich.com/553/how-to-chose-the-best-xml-parser-for-your-iphone-project 一文,基本比较准确和客观,文中建议:@H_502_3@
如果是读取很小的XML文档,性能基本上没有什么差别,不过从调用的方便性来说,建议使用TouchXML、KissXML或GdataxML@H_502_3@
如果是需要读取和修改XML文档,建议使用KissXML或GdataxML@H_502_3@
如果需要读取非常大的XML文档,则建议使用libxml2或TBXML@H_502_3@
如果你不想去调用第三方类库,那么使用NSXML也可以@H_502_3@
安装说明@H_502_3@
1. 从http://code.google.com/p/gdata-objectivec-client/downloads/list下载“gdata-objective-c client library.”@H_502_3@
2. 解压缩文件,找到Source\XMLSupport,并且将其中的GdataxMLNode.h 和 GdataxMLNode.m文件拖到项目中@H_502_3@
3. 选中项目,选中“Build Settings”标签页@H_502_3@
4. 将Build Settings页中,顶部的“Basic”标签切换到“All”@H_502_3@
5. 找到“Paths\Header Search Paths”项,并添加“/usr/include/libxml2”到列表中@H_502_3@
6. 找到“Linking\Other Linker Flags”项,并添加“-lxml2”到列表中@H_502_3@
在需要调用GdataxML的代码文件头部,加入:@H_502_3@
- #import“GdataxMLNode.h”
TouchXML@H_502_3@
1. 从https://github.com/TouchCode/TouchXML下载最新TouchXML源码@H_502_3@
2. 解压后,将其中的Sources目录下的文件和目录都拖到项目中@H_502_3@
@H_502_3@
在需要调用TouchXML的代码文件头部,加入:@H_502_3@
TBXML@H_502_3@
1. 从http://www.tbxml.co.uk/TBXML/TBXML_Free.html下载最新TBXML源码@H_502_3@
2. 解压后,将其中的TBXML.h,TBXML.m,NSDataAdditions.h和NSDataAddtions.m四个文件拖到项目中@H_502_3@
3. 添加对libz.dylib类库的应用@H_502_3@
在需要调用TBXML的代码文件头部,加入:@H_502_3@
使用说明@H_502_3@
读取XML文件,并解析成为DOM文档对象示例:@H_502_3@