關於XML解析的blog有很多,我本來不想寫的;不過我發現有一些細節他們都沒有說,我這裡就多說一些細節。
我們在哪些地方用XML:現在json用的這麼多,使用XML通訊的已經不多了。我遇到的場景是,我們的服務器有很多個,需要用戶去選擇。那麼我們就需要定期維護一個服務器列表,這個服務器列表的配置文件我需要每次下載。自然我就需要解析這一個文件。
XML解析有兩種模式SAX和DOM。我用的是系統的 NSXMLParser 它是SAX解析。
我寫了一個工具類,先貼一下代碼,下面會有一些說明
.h
#import <Foundation/Foundation.h> typedef NS_ENUM(NSInteger,xmlModelName){// 這裡我對我的XML文件做了區別 因為要解析表情和服務器列表兩種 xmlModelNameFace, xmlModelNameServer, }; @interface XMLParser : NSObject<NSXMLParserDelegate> // 這個是回調的Block @property(nonatomic,copy)void (^returnParseArray)(NSArray * returnArray); @property(nonatomic,readonly)xmlModelName currentModelName; - (instancetype)initWithFilePath:(NSString *)path fileType:(NSString *)fileType modelName:(xmlModelName)modelName; - (void)startWithFilePath:(NSString *)path fileType:(NSString *)fileType; @end
.m
#import "XMLParser.h" #import "FaceModel.h" #import "ToolClient.h" #import "ServerModel.h" @implementation XMLParser{ NSMutableArray * faceArray; NSMutableArray * serverArray; } @synthesize currentModelName; - (instancetype)initWithFilePath:(NSString *)path fileType:(NSString *)fileType modelName:(xmlModelName)modelName{ self = [super init]; if(self){ currentModelName = modelName; } return self; } - (void)startWithFilePath:(NSString *)path fileType:(NSString *)fileType { [self parseWithPath:path type:fileType]; } // - (void)parseWithPath:(NSString *)filePath type:(NSString *)fileType{ if (currentModelName == xmlModelNameFace) { faceArray = [[NSMutableArray alloc]init]; }else if(currentModelName == xmlModelNameServer){ serverArray = [[NSMutableArray alloc]init]; } NSData *xmlData = [[NSData alloc] initWithContentsOfFile:filePath]; if(xmlData && xmlData.length > 10){ NSXMLParser *parser = [[NSXMLParser alloc] initWithData:xmlData]; [parser setShouldProcessNamespaces:NO]; [parser setShouldReportNamespacePrefixes:NO]; [parser setShouldResolveExternalEntities:NO]; [parser setDelegate:self]; BOOL success = [parser parse]; if(success) { [self parseSuccess]; }else { [ToolClient activityShowMessage:@"XML解析失敗" inView:[UIApplication sharedApplication].windows[0]]; } }else { [ToolClient activityShowMessage:@"XML解析失敗" inView:[UIApplication sharedApplication].windows[0]]; } } // 成功後的回調 - (void)parseSuccess { if(self.returnParseArray){ if (currentModelName == xmlModelNameFace) { self.returnParseArray(faceArray); }else if(currentModelName == xmlModelNameServer){ self.returnParseArray(serverArray); } } } #pragma mark - NSXMLParserDelegate - (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qualifiedName attributes:(NSDictionary *)attributeDict { // NSLog(@"Name:%@",elementName); if([elementName isEqualToString:@"face"] && currentModelName == xmlModelNameFace) { FaceModel * faceModel = [[FaceModel alloc]init]; faceModel.kID = [attributeDict[@"id"]intValue]; faceModel.kName = attributeDict[@"name"]; faceModel.kImage = attributeDict[@"file"]; [faceArray addObject:faceModel]; faceModel = nil; }else if([elementName isEqualToString:@"Server"] && currentModelName == xmlModelNameServer){ ServerModel * sModel = [[ServerModel alloc]init]; sModel.serverName = attributeDict[@"name"]; sModel.serverIP = attributeDict[@"ChatServerIP"]; sModel.chatPort = attributeDict[@"chatPort"]; sModel.fileServerIP = attributeDict[@"fileServerIP"]; sModel.filePort = attributeDict[@"filePort"]; [serverArray addObject:sModel]; sModel = nil; } } - (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string { // NSLog(@"value:%@",string); } - (void)parserDidEndDocument:(NSXMLParser *)parser { // // NSLog(@"%@",faceArray); } - (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName{ // NSLog(@"elementName:%@",elementName); // NSLog(@"qualifiedName:%@",qName); // // NSLog(@"NSXMLParserDone"); // NSLog(@"%@",faceArray); // NSLog(@"%i",(int)faceArray.count); } @end
現在我來說一下XML解析的設置裡面那3個設置為NO的參數是什麼作用
[parser setShouldProcessNamespaces:NO]; [parser setShouldReportNamespacePrefixes:NO]; [parser setShouldResolveExternalEntities:NO];
第一個 setShouldProcessNamespaces 這個屬性設置為YES的話,這兩個方法會有值輸出:parser:didStartElement:namespaceURI:qualifiedName:attributes:
和 parser:didEndElement:namespaceURI:qualifiedName: 這兩個在解析過程中都是都是可以看到裡面的節點或字段的名字的。我覺得調試的時候可以用一下。
第二個 setShouldReportNamespacePrefixes 這個屬性設置為YES的話,這兩個方法會有值輸出:
parser:didStartMappingPrefix:toURI:
和 parser:didEndMappingPrefix: 這個我覺得完全沒有必要用它
第三個 setShouldResolveExternalEntities 這個屬性設置為YES的話,這個方法會有值輸出:parser:foundExternalEntityDeclarationWithName:publicID:systemID: 其中publicID 和systemID 都是XML文檔的特有的標識。官方文檔對這個兩個的變量都是這麼說的:
You may access this property once a parsing operation has begun or after an error occurs.
也就是當XML解析已經開始或者出現錯誤的時候,再去看它。也就是說如果你的XML寫的夠好 你就忽略它吧。
上面的代碼是工具類,下面這段會告訴你這段代碼怎麼用:
// 解析文件 - (void)parseFile:(NSString *)filepath{ NSLog(@"filepath%@",filepath);// 文件的路徑 if(data.length>10){// 簡單的長度檢測 __weak SelectServerViewController * ws = self;// 弱引用 // do parse XMLParser * xp = [[XMLParser alloc]initWithFilePath:filepath fileType:@"xml" modelName:xmlModelNameServer]; // 先設置回調 xp.returnParseArray = ^(NSArray * array){ // 回調的結果 去給tableView 展示 [ws gotDataArray:array]; }; // 再開始解析 [xp startWithFilePath:filepath fileType:@"xml"]; } } - (void)gotDataArray:(NSArray *)array { if(array){ // NSLog(@"array:%@",array); dataArray = [array mutableCopy]; [myTableView reloadData]; } }
XML解析我就寫這麼多了,給一個建議 解析的時候用一個Model來存儲數據,後續的使用會很方便。