iOS网络编程解析协议二:XML数据传输解析
XML两种解析方式,一种是SAX,NSXMLParser是SAX方法解析,另一种是DOM(Document Object Model);
区别:
SAX:
DOM:
#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数据传输解析的更多相关文章
- iOS网络编程解析协议三:JSON数据传输解析
作为一种轻量级的数据交换格式,正在逐步取代XML,成为网络数据的通用格式 基于JavaScript的一个子集 易读性略差,编码手写难度大,数据量小 JSON格式取代了XML给网络传输带来了很大的便利, ...
- iOS开发网络篇—网络编程基础(二)
下面叙述的是关于几个必须要知道的iOS网络编程入门级别的要点: 1.客户端如何找到连接的服务器 客户端通过URL找到想要连接的服务器 2.什么是URL URL的全称是Un ...
- iOS 网络编程(HTTP协议)
HTTP协议的概念HTTP协议,Hyper Text Transfer Protocol (超文本传输协议)是用于从万维网服务器传送超文本到本地浏览器的传输协议,HTTP是一个应用层协议,由请求和响应 ...
- iOS网络编程模型
iOS网络编程层次结构也分为三层: Cocoa层:NSURL,Bonjour,Game Kit,WebKit Core Foundation层:基于 C 的 CFNetwork 和 CFNetServ ...
- iOS网络编程笔记——Socket编程
一.什么是Socket通信: Socket是网络上的两个程序,通过一个双向的通信连接,实现数据的交换.这个双向连路的一端称为socket.socket通常用来实现客户方和服务方的连接.socket是T ...
- 网络编程——TCP协议、UDP协议、socket套接字、粘包问题以及解决方法
网络编程--TCP协议.UDP协议.socket套接字.粘包问题以及解决方法 TCP协议(流式协议) 当应用程序想通过TCP协议实现远程通信时,彼此之间必须先建立双向通信通道,基于该双向通道实现数 ...
- IOS网络编程——第三方类库
IOS网络编程——第三方类库 目录 概述 ASIHttpRequest AFNetworking 其他 概述 ASIHttpRequest AFNetworking 其他
- IOS网络编程:HTTP
IOS网络编程:HTTP HTTP定义了一种在服务器和客户端之间传递数据的途径. URL定义了一种唯一标示资源在网络中位置的途径. REQUESTS 和 RESPONSES: 客户端先建立一个TCP连 ...
- 网络编程TCP协议-聊天室
网络编程TCP协议-聊天室(客户端与服务端的交互); <span style="font-size:18px;">1.客户端发数据到服务端.</span> ...
随机推荐
- bzoj1941 hdu5992
看了青岛赛区的题简单学了一下kd,感觉这东西还是挺厉害的 一般kd树找最近点对最坏是O(n),但是随机情况下跑得还是很快的 kd树是一棵BST,但是每一层的关键字不同 一般写法是按照每一维轮流来,这一 ...
- Linux文件系统的详解
这里以 EXT2 文件系统为例 在Linux下,一个磁盘的最前面是MBR,大小为512Byte 在每一个分区下,第一部分是boot sector,接下来是super block,再接下来是inode, ...
- python开发学习-day02(元组、字符串、列表、字典深入)
s12-20160109-day02 *:first-child { margin-top: 0 !important; } body>*:last-child { margin-bottom: ...
- 自定义mvc验证特性,手机号号段老增加,给自定义一个RegularExpress
public class PhoneExpressionAttribute: RegularExpressionAttribute, IClientValidatable { public Phone ...
- 洛谷P3435 [POI2006]OKR-Period of Words [KMP]
洛谷传送门,BZOJ传送门 OKR-Period of Words Description 一个串是有限个小写字符的序列,特别的,一个空序列也可以是一个串. 一个串P是串A的前缀, 当且仅当存在串B, ...
- 简单模仿拉钩网上的“hot_info”
目录 前言 技术 判断进入div的方向 动画 绑定事件 css html 还需改进的地方 前言 突然想起来之前看到拉钩网上的hot_info(不知道该叫什么,但是拉钩网上这个div的class是hot ...
- HDU 6052 To my boyfriend(概率 贡献)
To my boyfriend Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)T ...
- Am335x u-boot 代码大概流程
在_面之前的流程和u-boot-spl一样,区别在于_main中. 对于u-boot 2016.03来说 ENTRY(_main) /* * Set up initial C runtime envi ...
- Web后门工具WeBaCoo
Web后门工具WeBaCoo WeBaCoo是使用Perl语言编写的Web后门工具.渗透测试人员首先使用该工具生成一个后门PHP页面.然后,将该页面上传到目标服务器上.最后,在本地终端直接访问该页 ...
- socketserver用法列子
socketserver socketserver内部使用IO多路复用以及“多线程”和“多进程”,从而实现并发处理多个客户端请求的scoket服务端.即,每个客户端请求连接到服务器时,socket服务 ...