iOS开发之关于网络数据的XML解析

关于XML:

网络应用中的数据解析,因为最近的应用,无论是Android的和ios平台的,一直用也是建议用的都是Json解析,xml解析都有点被遗忘了。

然后最近自己在做着玩一个ios的小应用,涉及网络数据的抓取,一些网站可能都提供了自己api平台,这些一般都是支持

我们对于数据协议格式的设定的。但是后来我在找寻到一些Rss资源时,发现返回的数据都是xml格式的,

因此,那就只好用xml解析了。

XML解析其实这个概念出现了算够久了,以前javaweb什么到处都在用。这边我们主要大致介绍下,然后在在ios编程如何使用。

XML解析一般分两种模式SAX和DOM,事件和文档。具体解析google去吧,有详细。不过看了下面的两个例子,一般就了解了.

一:XML解析之SAX解析,以及对NSXMLParser的应用

首先导入协议


XMLDemo.xml

 

    
        洛洛受
        
        14
    
    
        邹杰
        随神兽可随时变化
        17
    

让我们深入的了解SAX解析
首先定义一个可变数组用来盛放获取到的字符串

@property(nonatomic,retain)NSMutableString *mutableString; // 用来盛放获取的字符串
// XML的sax解析

- (void)xmlSaxParser{
    NSString *xmlPath = [[NSBundle mainBundle]pathForResource:@"XMLDemo.xml" ofType:nil];
    NSData *data = [NSData dataWithContentsOfFile:xmlPath];
    // 开始sax解析的学习
    NSXMLParser *xmlParser = [[NSXMLParser alloc]initWithData:data];
    // 由于sax解析是一个事件的处理过程,所以肯定是有顺序的,所以需要借助代理来执行解析过程
    xmlParser.delegate = self;
    // 开始解析, 返回值为BOOL类型,解析的过程是一个同步,意思就是说,只要开始解析,解析未完成,它后面的代码就不会执行。
    BOOL isParser = [xmlParser parse];
    if (isParser) {
        NSLog(@"解析成功");
    }else{
        NSLog(@"解析失败");
    }
    NSLog(@"我是洛洛不是裸裸");
}

由于sax解析是一个事件的处理过程,所以肯定是有顺序的,所以需要借助代理来执行解析过程,有5个代理,我自己写了一个方法用来 处理标签中的空格,换行, \t等制表符 (自己写的方法)

#pragma mark ----sax解析的代理方法
// 1.开始解析的时候,可以为容器进行初始化 (准备解析)
- (void)parserDidStartDocument:(NSXMLParser *)parser{
    self.mutableString = nil;
    NSLog(@"开始解析");
}

// 开始解析XML中的某个标签
/**
 *
 *
 *  @param parser        :解析器
 *  @param elementName   :节点名称
 *  @param namespaceURI  :命名空间中的URL
 *  @param qName         :命名空间的名称
 *  @param attributeDict :标签的属性
 */

// 2.准备解析节点
- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict{
    NSLog(@"开始解析标签-----%@---------%@",elementName,attributeDict);
}

// 3.从标签中取值  在foundCharacters:方法中一直保存当前最新的值
- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string{
    
  
    
    if (self.mutableString == nil) {
        self.mutableString = [NSMutableString stringWithString:string];
    }else{
        [self.mutableString appendString:string];
    }
    
    
    NSLog(@"从标签中取值------%@",string);
}


// 处理标签中的空格,换行, \t等制表符 (自己写的方法)
- (NSString *)replaceStringWithString:(NSMutableString *)string{
    NSString *str1 = [string stringByReplacingOccurrencesOfString:@"\n" withString:@""];
    NSString *str2 = [str1 stringByReplacingOccurrencesOfString:@" " withString:@""];
    NSString *str3 = [str2 stringByReplacingOccurrencesOfString:@"\r" withString:@""];
    NSString *str4 = [str3 stringByReplacingOccurrencesOfString:@"\t" withString:@""];
    
    
    return str4;
}




// 5.某个标签取值结束
- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName{
    // 根据标签名称来判断当前是哪个标签取值结束
    if ([elementName isEqualToString:@"name"]) { // 说明name取值结束了
        NSLog(@"name------%@",[self replaceStringWithString:self.mutableString]);
        
    }
    // 由于任意一个标签取值结束之后,可变字符串是重复利用的,所以需要重新置为nil,让它在取值代理方法中来保存
    self.mutableString = nil;
    
    NSLog(@"标签取值结束-------%@",elementName);
}

// 5.整个XML取值结束
- (void)parserDidEndDocument:(NSXMLParser *)parser{
    NSLog(@"整个取值结束");
}

//获取cdata块数据  
- (void)parser:(NSXMLParser *)parser foundCDATA:(NSData *)CDATABlock  
{  
//    NSLog(@"%@",NSStringFromSelector(_cmd) );  
}  

1.初始化解析器,传入你要解析的数据。
2.parse,启动解析,返回一个是否解析成功Bool值。

3.基本你要处理的就在下面实现的1-5个代理方法了。

其实代理方法和详细,就是一个事物进行流程:

step1是准备解析,然后没意外就是执行到了——>

step2读取到第一个头节点了,然后如果内部有属性值,你可以获取出来,读完头节点,我们会进去值域——》

step3对于简单的节点,可能直接就是一个string值了,但是看例子我们会知道,很多情况下,该节点的值域包含的于是一个节点——》

这步其实分两种,如果是值,那么就是执行step4,获取值的字符串,如果是子节点呢,我们一看就知道,它又是进行了step2,

即读取到头标签了,其实你是很人读一片文章流程一样,只不过我们脑中有个印象是头标签了,我们要做什么,读到头标签的最后一个符号">"

下面进去值域,读到了字符串的话就调用了foundCharacters:(NSString *)string,如果又读到这样的,那就又是头标签了。——》

step5就是读到开始尾标签符号了。

最后一个方法

foundCDATA:(NSData *)CDATABlock,其实也是一个格式

二:Dom文档解析模型,TBXML第三方包应用。(需要导入一个第三方的头文件)

dom解析模型就像一个树结构,节点,子节点,兄弟节点等等。
这个其实最后被我抛弃了,这个解析器太简化了,太简洁的东西导致控制的入口点太少,就比如一个一键优化的软件的概念是一样的,
一键清楚缓存,优化配置,文件归类等等。人为控制就少了,导致我解析上面那个模型时,只知道遍历存储~。但是这个解析期对部分要求不高的xml解析其实挺好分,真的很简洁。

#import "GDataXMLNode.h"
#pragma mark - XML的dom解析方式
// XML的dom解析方式
- (void)xmlDomParser{
    NSString *xmlPath = [[NSBundle mainBundle]pathForResource:@"XMLDemo.xml" ofType:nil];
    NSData *data = [NSData dataWithContentsOfFile:xmlPath];
    
    
  // 将XML文件读取到内存中,并转为文档模型,并且是树状结构
    NSError *error;
    GDataXMLDocument *xmlDocument = [[GDataXMLDocument alloc]initWithData:data options:0 error:&error];
     // dom解析的时候,从根节点开始解析,如果只有一个节点,那么咱们就直接取值,如果此节点还有子节点,那么就一直找寻子节点,直到找到值为止。找寻过程肯定是由外而内,也就是从根节点开始,一直往树状结构的底部查询
    // 得到根节点
    GDataXMLElement *rootElement = [xmlDocument rootElement];
    NSLog(@"根节点的标签----%@",rootElement.name);
    
    
    // 在添加一个学生节点
    // 创造一个student节点
    GDataXMLElement *createStudentNode = [GDataXMLElement elementWithName:@"student"];
    // 为student节点创造子节点name,age,sex
    GDataXMLElement *createNameNode = [GDataXMLElement elementWithName:@"name" stringValue:@"大定位"];
    GDataXMLElement *createAgeNode = [GDataXMLElement elementWithName:@"age" stringValue:@"12"];
    GDataXMLElement *createSexNode = [GDataXMLElement elementWithName:@"sex" stringValue:@"男"];
    
    // 为student节点添加刚才创建好的子节点
    [createStudentNode addChild:createNameNode];
    [createStudentNode addChild:createAgeNode];
    [createStudentNode addChild:createSexNode];
    // 为根节点添加刚才创建好的student节点
    [rootElement addChild:createStudentNode];
    
    
    
    // 得到根节点底部的子节点 由于每个节点的子节点都有可能是多个,所以返回值为数组。所有子节点的获得,都是他们的父节点通过节点名称(标签名称)得到所有的子节点。
    NSArray *subElement = [rootElement elementsForName:@"student"];
    NSLog(@"%@",subElement);
    // 取出student节点的子节点
    for (GDataXMLElement *element in subElement) {
        // 此处的element是某一个student节点
        // 取出name节点 他是student的子节点
        NSArray *nameElement = [element elementsForName:@"name"];
        // 由于name节点没有子节点,节点中的值是我们所需要的。并且我们知道nameElement数组中只有一个元素。并且他的类型是GDataElement。但是我们知道该节点中的值的类型为字符串,所以下面需要强制转换。
        // 取值第一步:取出上面数组中的元素
        GDataXMLElement *nameValue = [nameElement objectAtIndex:0];
        // 第二步:强制转换
        NSString *name = [nameValue stringValue];
        NSLog(@"nameValue-----%@",name);
    }
}

最后在viewDidLoad中调用方法即可。
- (void)viewDidLoad {
    [super viewDidLoad];
//    [self xmlDomParser];
    [self xmlSaxParser];
}

你可能感兴趣的:(iOS开发之关于网络数据的XML解析)