iOS SDK中使用NSXMLParser解析XML(iphone網絡篇三)

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];
複製代碼

固然還能夠有其餘初始化生成方法,如:post

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回調函數中完成。url

複製代碼
//開始解析前,在這裏能夠作一些初始化工做
// 假設已聲明有實例變量 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 {
//記錄所取得的文字列
}

- (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 {
//.....
}


有關我學習NSXMLPareser解析xml曾經產生的問題與解疑:spa

  •  相似以下XML文檔結構,其分析是從imgList開始讀取,而不是img表節點。否則就只讀取一次,是不會循環的。
    <imgList><img><src>圖片地址1</src><name>圖片名稱1</name><url>圖片指定超連接1</url></img>
    <img><src>圖片地址2</src><name>圖片名稱2</name><url>圖片指定超連接2</url></img></imgList> 
  • 在根據xml文檔結構組織相關實體類定義時的基本思路:把每一個新節點容器都定義描述成一個新的實體類。
相關文章
相關標籤/搜索