iOS SDK的NSXMLParser解析XML文檔是事件驅動模式的,即采用SAX方式來解析XML格式文檔。NSXMLParser在處理XML文檔的過程中當遇到一些要素(元素、屬性、CDATA塊、評論等)時會通知它的委托,而自身不對解析的要素進行任何處理,全權委托給NSXMLParserDelegate處理。同時它也會報告錯誤。
1. 打開一個xml文件,讀取內容到NSData中。
NSString *path = [[NSBundle mainBundle] pathForResource:@"filename" ofType:@"xml"];
NSFileHandle *file = [NSFileHandle fileHandleForReadingAtPath:path];
NSData *data = [file readDataToEndOfFile];
[file closeFile];
2. 調用NSXMLParser的initWithData:方法,並設置代理delegate。
NSXMLParser *m_parser = [[NSXMLParser alloc] initWithData:data];
//設置該類本身為代理類,即該類在聲明時要實現NSXMLParserDelegate委托協議
[m_parser setDelegate:self]; //設置代理為本地
BOOL flag = [m_parser parse]; //開始解析
if(flag) {
NSLog(@"獲取指定路徑的xml文件成功");
}else{
NSLog(@"獲取指定路徑的xml文件失敗");
}
[m_parser release];
當然還可以有其他初始化生成方法,如:
NSXMLParser *parser = [[NSXMLParser alloc] initWithContentsOfURL:[NSURL URLWithString:urlString]];
直接自定義一個方法來實現創建解析:
1 - (void)parseXMLFileAtURL:(NSURL *)URL parseError:(NSError **)error{
2 NSXMLParser *parser = [[NSXMLParser alloc] initWithContentsOfURL:URL];
3 [parser setDelegate:self];
4 [parser setShouldProcessNamespaces:NO];
5 [parser setShouldReportNamespacePrefixes:NO];
6 [parser setShouldResolveExternalEntities:NO];
7 [parser parse];
8 NSError *parseError = [parser parserError];
9 if (parseError && error) {
10 *error = parseError;
11 }
12 [parser release];
13 }
3. 調用回調函數。
當完成上面的parser初始化並執行parser語句時([parser parse]),程序就會跳到代理方法,調用回調函數didStartElement,該方法會將整個xml遍歷一遍,並識別xml裡面的元素名稱(elementName),在發現要查找的信息時,創建一個如數組或其他變量以便在合適的時候存儲這些信息。一般地,具體完成相關存儲的操作往往在didEndElement回調函數中完成。
//開始解析前,在這裡可以做一些初始化工作
// 假設已聲明有實例變量 dataDict,parserObject
- (void)parserDidStartDocument:(NSXMLParser *)parser {
dataDict = [[NSMutableDictionary alloc] initWithCapacity:0]; //每一條信息都用字典來存儲
parserObjects = [[NSMutableArray alloc] init]; //每一組信息都用數組來存,最後得到的數據即在此數組中
}
//當解析器對象遇到xml的開始標記時,調用這個方法。
//獲得結點頭的值
//解析到一個開始tag,開始tag中可能會有properpies,例如<book catalog="Programming">
//所有的屬性都存儲在attributeDict中
- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qualifiedName attributes:(NSDictionary *)attributeDict {
if([elementName isEqualToString:@"book"]) {
NSString *catalog = [attributeDict objectForKey:@"catalog"];
}else if() {
//......
}
}
//當解析器找到開始標記和結束標記之間的字符時,調用這個方法。
//解析器,從兩個結點之間讀取具體內容
- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string {
//記錄所取得的文字列
}www.2cto.com
- (void)parser:(NSXMLParser *)parser foundCDATA:(NSData *)CDATABlock{
//NSLog(@"cData:%@",[NSString stringWithUTF8String:[CDATABlock bytes]]);
}
//當解析器對象遇到xml的結束標記時,調用這個方法。
//獲取結點結尾的值,此處為一Tag的完成
- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName {
}
//xml解析結束後的一些操作可在此
- (void)parserDidEndDocument:(NSXMLParser *)parser {
//.....
}