XML两种解析方式,一种是SAX,NSXMLParser是SAX方法解析,另一种是DOM(Document Object Model);

区别:

SAX:

只能读,不能修改,只能顺序访问,适合解析大型XML,解析速度快
常应用于处理大量数据的XML,实现异构系统的数据访问,实现跨平台
从文档的开始通过每一节点移动,定位一个特定的节点

DOM:

不仅能读,还能修改,而且能够实现随机访问,缺点是解析速度慢,适合解析小型文档
一般应用与小型的配置XML,方便操作
为载入到内存的文档节点建立类型描述,呈现可横向移动、潜在巨大的树型结构
在内存中生成节点树操作代价昂贵
 
这里我们主要介绍使用SAX方式,NSXMLParser是SAX的方法解析,NSXMLParser解析通过NSXMLParserDelegate协议代理方法来完成,解析过程如下:
NSXMLParser解析过程:
<1>. 创建NSXMLParser实例,并传入从服务器接收的XML数据
<2>. 定义解析器代理
<3>. 解析器解析
<4.. 通过解析代理方法完成XML数据的解析
 
举例一:把本机服务器上的student.xml文件中数据解析并输出到控制台上。
文件和内容截图截图如下:
      
 
具体代码如下:
//创建一个学生Student类,并声明属性:姓名name、年龄age、性别sex
 #import <Foundation/Foundation.h>

 @interface Student : NSObject
@property (copy,nonatomic)NSString *name;
@property (assign,nonatomic)NSInteger age;
@property (assign,nonatomic)char sex;
@end

//重写输出方法

 #import "Student.h"

 @implementation Student

 //重写输出方式
-(NSString*)description
{
return [NSString stringWithFormat:@"name:%@,age:%ld,sex:%c",_name,_age,_sex];
} @end

//在ViewController类中带入头文件和设置相关的属性信息,同时实现解析协议<NSXMLParserDelegate>

 #import "ViewController.h"
#import "Student.h" @interface ViewController ()<NSXMLParserDelegate>
@property (strong,nonatomic)NSMutableArray *stus; //存储解析到的学生对象
@property (strong,nonatomic)Student *currentStudent; //解析到的当前学生对象
@property (strong,nonatomic)NSMutableString *elementValue; //节点值 @end

//下载数据并进行解析

 - (void)viewDidLoad
{
[super viewDidLoad]; //1.从网络下载数据
NSURL *URL = [NSURL URLWithString:@"http://localhost/userManager/student.xml"];
NSData *data = [NSData dataWithContentsOfURL:URL]; //2.创建解析对象
NSXMLParser *parser = [[NSXMLParser alloc]initWithData:data]; //3.设置解析对象的代理
parser.delegate = self; //4.开始解析
[parser parse];
}

//实现代理的解析方法

#pragma mark -<NSXMLParserDelegate>

//文档开始
 //文档开始
-(void)parserDidStartDocument:(NSXMLParser *)parser
{
//初始化数组
self.stus = [NSMutableArray array];
}
 
//文档结束
 //文档结束
-(void)parserDidEndDocument:(NSXMLParser *)parser
{
//输出解析后得到的数据
[self.stus enumerateObjectsUsingBlock:^(Student *student, NSUInteger idx, BOOL *stop) {
NSLog(@"%@",student);
}];
}
 
//元素开始
 //元素开始
-(void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict
{
//创建学生对象
if([elementName isEqualToString:@"student"])
{
self.currentStudent = [[Student alloc]init];
} //初始化节点值
else if ([elementName isEqualToString:@"name"] ||
[elementName isEqualToString:@"age"] ||
[elementName isEqualToString:@"sex"] )
{
self.elementValue = [[NSMutableString alloc]init];
}
}
//找到文字
 //找到文字
-(void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string
{
//获取节点值
[self.elementValue appendString:string];
}
//元素结束
 //元素结束
-(void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName
{
//解析得到学生对象姓名
if([elementName isEqualToString:@"name"])
{
self.currentStudent.name = self.elementValue;
} //解析得到学生对象年龄
else if([elementName isEqualToString:@"age"])
{
self.currentStudent.age = [self.elementValue integerValue];
} //解析得到学生对象性别
else if([elementName isEqualToString:@"sex"])
{
self.currentStudent.sex = [self.elementValue characterAtIndex:];
} //将学生对象添加到数组中
else if ([elementName isEqualToString:@"student"])
{
[self.stus addObject:self.currentStudent];
}
}
//解析出错
 //解析出错
-(void)parser:(NSXMLParser *)parser parseErrorOccurred:(NSError *)parseError
{
NSLog(@"解析出错:%@",parseError);
}

输出结果如下:

-- ::34.377 XMLParser解析student.xml[:] name:kellen,age:,sex:M
-- ::34.378 XMLParser解析student.xml[:] name:jack,age:,sex:M

举例二:把本机服务器上的cities.xml文件中数据解析并输出到控制台上,这个比较复杂一些。

文件截图和部分内容截图(内容太多,不便全部截图)如下

  

具体代码如下:都在ViewController类中写代码

//实现协议<NSXMLParserDelegate>并声明必要的属性

 #import "ViewController.h"

 @interface ViewController ()<NSXMLParserDelegate>
@property (strong,nonatomic)NSMutableDictionary *province; //存储所有的省份
@property (strong,nonatomic)NSMutableDictionary *city; //存放对应省份的所有城市
@property (strong,nonatomic)NSMutableString *keyName; //字典的键名
@property (strong,nonatomic)NSMutableString *elementValue; //节点的值
@property (strong,nonatomic)NSMutableArray *arrayTemp; //临时数组,存放省名或城市名
@end

//下载数据并进行解析

 - (void)viewDidLoad
{
[super viewDidLoad]; //1.从网络下载数据
NSURL *URL = [NSURL URLWithString:@"http://localhost/userManager/cities.xml"];
NSData *data = [NSData dataWithContentsOfURL:URL]; //2.创建解析对象
NSXMLParser *parser = [[NSXMLParser alloc]initWithData:data]; //3.设置解析对象的代理
parser.delegate = self; //4.开始解析
[parser parse];
}

//实现解析代理的方法

#pragma mark -<NSXMLParserDelegate>

文档开始

 //文档开始
-(void)parserDidStartDocument:(NSXMLParser *)parser
{
//初始化省份字典
self.province = [NSMutableDictionary dictionary];
}

文档结束

 //文档结束
-(void)parserDidEndDocument:(NSXMLParser *)parser
{
//输出所有的城市
[self.province setObject:self.city forKey:@"cities"];
NSLog(@"%@",self.province);
}

元素开始

 //元素开始
-(void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict
{
//初始化临时数组
if([elementName hasPrefix:@"array"])
{
self.arrayTemp = [NSMutableArray array];
} //初始化节点值
else if ([elementName hasPrefix:@"key"] || [elementName hasPrefix:@"string"])
{
self.elementValue = [[NSMutableString alloc]init];
} //初始化城市字典
else if ([elementName hasPrefix:@"dict"])
{
if ([self.keyName hasPrefix:@"cities"])
{
self.city = [NSMutableDictionary dictionary];
}
}
}

找到文字

 //找到文字
-(void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string
{
[self.elementValue appendString:string];
}

元素结束

 //元素结束
-(void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName
{
//取节点值
if([elementName hasPrefix:@"key"])
{
self.keyName = self.elementValue;
} //将省份或城市添加到临时数组
else if ([elementName hasPrefix:@"string"])
{
[self.arrayTemp addObject:self.elementValue];
} //将临时数组添加到字典中
else if([elementName hasPrefix:@"array"])
{
if([self.keyName hasPrefix:@"provinces"])
{
[self.province setObject:self.arrayTemp forKey:self.keyName];
}
else
{
[self.city setObject:self.arrayTemp forKey:self.keyName];
}
}
}

解析出错

 //解析出错
-(void)parser:(NSXMLParser *)parser parseErrorOccurred:(NSError *)parseError
{
NSLog(@"解析出错:%@",parseError);
}

运行结果如下:(由于编码的不同,无法显示汉字,但是结果是正确的,同样内容太多,只给出部分结果)

-- ::35.139 XMLParse解析cities.xml[:] {
cities = {
"\U4e0a\U6d77\n\t\t\n\t\t\t" = (
"\U4e0a\U6d77\n\t\t\n\t\t"
);
"\U4e91\U5357\n\t\t\n\t\t\t" = (
"\U6606\U660e\n\t\t\t",
"\U4fdd\U5c71\n\t\t\t",
"\U695a\U96c4\n\t\t\t",
"\U5927\U7406\n\t\t\t",
"\U5fb7\U5b8f\n\t\t\t",
"\U8fea\U5e86\n\t\t\t",
"\U4e2a\U65e7\n\t\t\t",
"\U4e3d\U6c5f\n\t\t\t",
"\U4e34\U6ca7\n\t\t\t",
"\U6012\U6c5f\n\t\t\t",
"\U66f2\U9756\n\t\t\t",
"\U601d\U8305\n\t\t\t",
"\U6587\U5c71\n\t\t\t",
"\U897f\U53cc\U7248\U7eb3\n\t\t\t",
"\U7389\U6eaa\n\t\t\t",
"\U662d\U901a\n\t\t\n\t\t"
);
"\U5185\U8499\U53e4\n\t\t\n\t\t\t" = (
"\U547c\U548c\U6d69\U7279\n\t\t\t",
"\U963f\U62c9\U5584\U76df\n\t\t\t",
"\U5df4\U5f66\U6dd6\U5c14\U76df\n\t\t\t",
"\U5305\U5934\n\t\t\t",
"\U8d64\U5cf0\n\t\t\t",
"\U9102\U5c14\U591a\U65af\n\t\t\t",
"\U547c\U4f26\U8d1d\U5c14\n\t\t\t",
"\U901a\U8fbd\n\t\t\t",
"\U4e4c\U6d77\n\t\t\t",
"\U4e4c\U5170\U5bdf\U5e03\U76df\n\t\t\t",
"\U9521\U6797\U90ed\U52d2\U76df\n\t\t\t",
......................................
......................................

iOS网络编程解析协议二:XML数据传输解析的更多相关文章

  1. iOS网络编程解析协议三:JSON数据传输解析

    作为一种轻量级的数据交换格式,正在逐步取代XML,成为网络数据的通用格式 基于JavaScript的一个子集 易读性略差,编码手写难度大,数据量小 JSON格式取代了XML给网络传输带来了很大的便利, ...

  2. iOS开发网络篇—网络编程基础(二)

    下面叙述的是关于几个必须要知道的iOS网络编程入门级别的要点:       1.客户端如何找到连接的服务器    客户端通过URL找到想要连接的服务器   2.什么是URL     URL的全称是Un ...

  3. iOS 网络编程(HTTP协议)

    HTTP协议的概念HTTP协议,Hyper Text Transfer Protocol (超文本传输协议)是用于从万维网服务器传送超文本到本地浏览器的传输协议,HTTP是一个应用层协议,由请求和响应 ...

  4. iOS网络编程模型

    iOS网络编程层次结构也分为三层: Cocoa层:NSURL,Bonjour,Game Kit,WebKit Core Foundation层:基于 C 的 CFNetwork 和 CFNetServ ...

  5. iOS网络编程笔记——Socket编程

    一.什么是Socket通信: Socket是网络上的两个程序,通过一个双向的通信连接,实现数据的交换.这个双向连路的一端称为socket.socket通常用来实现客户方和服务方的连接.socket是T ...

  6. 网络编程——TCP协议、UDP协议、socket套接字、粘包问题以及解决方法

    网络编程--TCP协议.UDP协议.socket套接字.粘包问题以及解决方法 TCP协议(流式协议) ​ 当应用程序想通过TCP协议实现远程通信时,彼此之间必须先建立双向通信通道,基于该双向通道实现数 ...

  7. IOS网络编程——第三方类库

    IOS网络编程——第三方类库 目录 概述 ASIHttpRequest AFNetworking 其他 概述 ASIHttpRequest AFNetworking 其他

  8. IOS网络编程:HTTP

    IOS网络编程:HTTP HTTP定义了一种在服务器和客户端之间传递数据的途径. URL定义了一种唯一标示资源在网络中位置的途径. REQUESTS 和 RESPONSES: 客户端先建立一个TCP连 ...

  9. 网络编程TCP协议-聊天室

    网络编程TCP协议-聊天室(客户端与服务端的交互); <span style="font-size:18px;">1.客户端发数据到服务端.</span> ...

随机推荐

  1. java EE :Servlet 接口

    Servlet 生命周期  : 调用当前 Servlet 类构造函数进行实例化 Servlet 通过调用 init () 方法进行初始化 Servlet 调用 service() 方法来处理客户端的请 ...

  2. C# 6.0 新特性 (四)

    原文: 1.http://www.cnblogs.com/BoyceYang/p/3711343.html 2.http://www.cnblogs.com/lhking/p/3660182.html ...

  3. Robot Framework + Selenium2Lib

    Robot Framework + Selenium2Lib 最近一段时间,公司在推行自动化测试流程,本人有幸参与了自定义通用控件的关键字封装和脚本辅助编写.数据驱动管理.测试用例执行管理等一系列工具 ...

  4. jquery通配符说明

    按姓名匹配 1,name前缀为aa的所有div的jquery对象 Js代码 收藏代码$("div[name^='aa']"); 2,name后缀为aa的所有div的jquery对象 ...

  5. 一个简单的ajax上传 上传进度显示

    本例用了jquery.form.js请到演示页面查看 CSS Code <style> form { display: block; margin: 20px auto; backgrou ...

  6. thinkphp5.0Traits引入

    ThinkPHP 5.0开始采用trait功能(PHP5.4+)来作为一种扩展机制,可以方便的实现一个类库的多继承问题. Traits 是一种为类似 PHP 的单继承语言而准备的代码复用机制.Trai ...

  7. java.io.writer API 以及 源码解读

    声明 我看的是java7的API文档. 如下图所示,java.io.writer 继承了java.lang.Object,实现的接口有Closeable, Flushable, Appendable, ...

  8. NumPy简明教程(二、数组1)

    NumPy数组 NumPy数组是一个多维数组对象,称为ndarray.其由两部分组成: 实际的数据 描述这些数据的元数据 大部分操作仅针对于元数据,而不改变底层实际的数据. 关于NumPy数组有几点必 ...

  9. BZOJ4554 HEOI2016游戏

    网络流. 我一开始傻傻的将每条边与每一列连边,边权为联通块树,但是这样做是错的因为我们就在跑网络流中会忽略掉边和列的关系. 我们在做网络流题时一定要注意题目中给出的限制条件,一定要以限制条件建图!!! ...

  10. 背包的第k优解[动态规划]

    From easthong ☆背包的第k优解                 描述 Description     DD 和好朋友们要去爬山啦!他们一共有 K 个人,每个人都会背一个包.这些包的容量是 ...