--- XML ---
1.可扩展标记语言(eXtensible Markup Language)
2.跟JSON解析一样,常用于交互的数据格式
3.一般XML文档(XML Document)
4.举例
<student>【标签,表示开始 /*】
<name>你</name>
<hobby>女</hobby>
</student> 【表示结束】
5.XML文档的语法:
5.1 文档声明:在XML文档最前面,必须写一个文档声明。
- ①最简单的声明:<?xml version="1.0"?">
- ②用encoding属性,来说明文档字符编码:<?xml version="1.0" encoding="UTF-8"?>
5.2 元素(Element):一个元素,会包含一个开始标签和一个结束标签
- ①有元素内容:<name>你</name>
- ②无元素内容:<name></name>
而且一个元素可嵌套若干个元素,但不能出现交叉嵌套
<students>
<student>
<name>你</name>
<hobby>女</hobby>
</student>
</students>```
#### 注意:
* 规范的XML文档最多只有一个根元素,其他的都是根元素的子孙元素。
* 在XML文档中,空格和换行,都会被当做具体内容去处理,例如:
```code①和②是不一样的
①<name>你</name>
②<name>
你
</name>```
5.3 属性(Atribute):一个元素可以拥有多个属性,例如:
<video name="小黄人" hobby="banana"/>【video元素拥有name,hobby两个属性】
6.对XML文档的解析;
提取```name```元素里面的内容:```<name>小黄人</name>```
或者是提取```video```元素中```name```和```hobby```的值:```<video name="小黄人" hobby="banana">```
>--- SAX解析 ---
```code
<students>
<student>
<name>你</name>
<hobby>女</hobby>
</student>
</students>```
1.解析方式,逐行进行解析。
1.1 打开XML文档
1.2 开始节点(开始标签)
1.2.1 <students>
1.2.2 <student>
1.2.3 <name>
1.3 发现元素里面的内容
1.3.1 提取到”你“
1.4 结束节点(结束标签)
1.4.1 </name>
1.5 循环开始,直到遇到</students>解析才彻底结束
2.iOS中XML解析方式有很多,
2.1 苹果原生:NSXMLParser-–SAX方式解析,使用简单。
2.2 第三方框架:
①libxml2:纯C语言,默认被包含进了iOS SDK,同时支持DOM和SAX解析
②GDataXML:DOM的解析方式,是由谷歌开发的,底层基于libxml2```
1.XML_SAX解析(下边就上边的students例子作简单说明下SAX解析)
#import "FirstViewController.h"
#import "Student.h"
@interface FirstViewController ()<NSXMLParserDelegate>//遵循协议
/* 数据源数组 */
@property (strong,nonatomic) NSMutableArray *dataArray;
/* 当前标签 */
@property (copy,nonatomic) NSString *currentElement;
@end
@implementation FirstViewController
- (void)viewDidLoad {
[super viewDidLoad];
[self SAXParser];
}
-(void)SAXParser{
//1.获取文件路径
NSString *path = [[NSBundle mainBundle] pathForResource:@"XML" ofType:@"txt"];
//2.通过路径,创建一个data对象
NSData *data = [NSData dataWithContentsOfFile:path];
//3.创建SAX解析使用的对象
NSXMLParser *xmlParser = [[NSXMLParser alloc] initWithData:data];
//4.遵循代理
xmlParser.delegate = self;
//5.开始解析
[xmlParser parse];
}
#pragma mark --- NSXMLParserDelegate方法 ---
//1.开始解析XML文档
- (void)parserDidStartDocument:(NSXMLParser *)parser{
//准备数据(自动根据数组元素个数开辟空间,最好先给0)
self.dataArray = [NSMutableArray arrayWithCapacity:0];
}
//2.开始解析标签(节点)
/*
parser:类
elementName:标签名
namespaceURI:节点里的命名空间
*/
- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary<NSString *,NSString *> *)attributeDict{
//获取到当前正在解析的标签,给标签属性赋值
_currentElement = elementName;
if ([elementName isEqualToString:@"student"]) {
#warning 创建并初始化model,如果标签是“student”那么就创建model类,并且将model添加到数据源数组中(model的预建立)
Student *stu = [[Student alloc] init];
[self.dataArray addObject:stu];
}
}
//3.循环读取标签的值:这个方法,得到节点里面的存的信息数据
- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string{
//将值存入,model类对应的属性中,这个方法循环执行,我们需要取的值是最后一个【包<student></student>,是按照顺序扔进数组的,挨个进行取值,所以只取最后一个,说明全部取完】
Student *stu = self.dataArray.lastObject;
//根据当前解析数来的标签取到取到跟stu中属性对应的值,赋值给stu中的属性
[stu setValue:string forKey:_currentElement];
}
//4. 结束标签解析
- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName{
//没有标签,解析结束了
_currentElement = nil;
}
//5.结束解析文档
- (void)parserDidEndDocument:(NSXMLParser *)parser{
for (Student *stu in self.dataArray) {
NSLog(@"%@",stu);
}
}
//6.错误报告
- (void)parser:(NSXMLParser *)parser parseErrorOccurred:(NSError *)parseError{
NSLog(@"错误报告:%@",parseError);
}
@end
2.DOM(Document Object Model)解析
1.Document object Model(文档对象模型),将整个XML文档读入,构建驻留内存的树结构(节点数),通过遍历树结构的任意节点,读取他的属性和值
2:步骤
2.1:获取GDataXMLNode.h文件,将GDataXMLNode.m添加到所需工程
2.2:添加类库libxml2.tbd到工程中
2.3:在工程中的"Build settings"页面中找到"Header Search path"项,添加"/usr/include/libxml2"
3.XML_DOM解析
#import "SecondViewController.h"
@interface SecondViewController ()
/* 数据源数组 */
@property (strong,nonatomic) NSMutableArray *dataArray;
@end
@implementation SecondViewController
- (void)viewDidLoad {
[super viewDidLoad];
[self XML_DOM];
}
-(void)XML_DOM{
//1.获取文件路径
NSString *path = [[NSBundle mainBundle]pathForResource:@"XML" ofType:@"txt"];
//2.以NSData类型进行分析,将文件读入到data中
NSData *data = [NSData dataWithContentsOfFile:path];
//3.创建
GDataXMLDocument *xmlDocument = [[GDataXMLDocument alloc] initWithData:data options:0 error:nil];
//4.创建根节点对象,获取XML文档的根节点里包含了XML文档中的所有内容。
GDataXMLElement *RootElement = [xmlDocument rootElement];
//5.创建解析出来的学生数组
self.dataArray = [NSMutableArray array];
//遍历根节点<students>的所有的子节点<student>
for (GDataXMLElement *subElement in RootElement.children) {
Student *stu = [[Student alloc] init];
//再遍历子节点<student>下的子标签<name><zi><say><weapon>,取出子标签的内容,赋值给stu
for (GDataXMLElement *contactElement in subElement.children) {
if ([contactElement.name isEqualToString:@"name"]) {
stu.name = contactElement.stringValue;
}else if ([contactElement.name isEqualToString:@"zi"]) {
stu.zi = contactElement.stringValue;
}else if ([contactElement.name isEqualToString:@"say"]) {
stu.say = contactElement.stringValue;
}else if ([contactElement.name isEqualToString:@"weapon"]) {
stu.weapon = contactElement.stringValue;
}
//使用KVC对stu整个赋值
[stu setValue:contactElement.stringValue forKeyPath:contactElement.name];
}
//将赋值后的model类放入数据源数组
[self.dataArray addObject:stu];
}
//打印结果
for (Student *stu in self.dataArray) {
NSLog(@"%@",stu);
}
}
@end