前段时间想找点事做,就是试着看能不能用豆瓣的API做点什么,于是就碰到了这个问题——XML解析。
老师还没讲,只能自己去查。
XML文档解析主要有SAX和DOM两种模式,IOS上两种模式都可以用,这里就不做过多介绍,我选择的SAX模式。
IOS解析XML用的是自带的NSXML框架,框架的核心是NSXMLParser类和它的委托协议NSXMLParserDelegate,其主要的解析工作是在NSXMLParserDelegate实现类中完成的。委托中定义了许多回掉方法,在SAX解析器从上到下遍历XML文档的过程中,遇到开始标签、结束标签、文档开始、文档结束和字符串结束是就会触发这些方法。这些方法有很多,下面我们列出5个常用的方法。
在文档开始时触发
-(void)parserDidStartDocument:(NSXMLParser *)parser
遇到一个新标签是触发,其中namespaceURI是命名空间,qualifiedName是限定名,attributes是字典类型的属性集合。
-(void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict
找到字符串时触发
-(void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string
遇到结束标签时触发
-(void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName
在文档结束时触发
-(void)parserDidEndDocument:(NSXMLParser *)parser
下面通过一个具体的例子来看整个的调用与解析过程
首先这是我们将要解析的XML文件 "info.xml"
<?xml version="1.0" encoding="UTF-8"?> <root> <person id="1"> <firstName>Wythe</firstName>
<lastName>xu</lastName>
<age>22</age> </person>
<person id="2">
<firstName>li</firstName>
<lastName>si</lastName>
<age>31</age>
</person>
<person id="3">
<firstName>Dipen</firstName>
<lastName>Shah</lastName>
<age>24</age>
</person>
</root>
接来来是一个头文件 "ViewController.h"
#import <UIKit/UIKit.h> @interface ViewController : UIViewController<NSXMLParserDelegate> @property NSXMLParser *parser;
@property NSMutableArray *person;
@property NSString *currenttag; @end
然后是它的实现文件 "ViewController.m"
#import "ViewController.h" @interface ViewController () @end @implementation ViewController @synthesize parser = _parser , person = _person , currenttag = _currenttag; - (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
if (self) {
// Custom initialization
}
return self;
} - (void)viewDidLoad
{
[super viewDidLoad]; NSString *xmlFilePath = [[NSBundle mainBundle]pathForResource:@"info"ofType:@"xml"]; NSData *data = [[NSData alloc]initWithContentsOfFile:xmlFilePath]; self.parser = [[NSXMLParser alloc]initWithData:data]; self.parser.delegate = self; [self.parser parse]; NSLog(@"%@",_person); } - (void)didReceiveMemoryWarning
{
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
} #pragma mark delegate method -(void)parserDidStartDocument:(NSXMLParser *)parser
{
_person = [[NSMutableArray alloc]init];
NSLog(@"start parse 1");
} -(void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict
{
_currenttag = elementName; if ([_currenttag isEqualToString:@"person"]) {
NSString *_id = [attributeDict objectForKey:@"id"];
NSMutableDictionary *dict = [[NSMutableDictionary alloc]init];
[dict setObject:_id forKey:@"id"];
[_person addObject:dict];
} NSLog(@"start element");
} -(void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string
{
NSMutableDictionary *dict = [_person lastObject]; if ([_currenttag isEqualToString:@"firstName"] && dict) {
[dict setObject:string forKey:@"firstName"]; }
if ([_currenttag isEqualToString:@"lastName"] && dict) {
[dict setObject:string forKey:@"lastName"];
}
if ([_currenttag isEqualToString:@"age"] && dict) {
[dict setObject:string forKey:@"age"];
} NSLog(@"found characters");
} -(void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName
{
_currenttag = nil; NSLog(@"end element");
} -(void)parserDidEndDocument:(NSXMLParser *)parser
{
NSLog(@"parse end");
}
@end
通过断电和输出信息,我们可以知道整个解析过程是 开始解析文档、开始标签、找到字符串、结束标签、文档结束。
-- ::32.920 xmlforblog[:60b] start parse
-- ::32.921 xmlforblog[:60b] start element
-- ::32.922 xmlforblog[:60b] found characters
-- ::32.922 xmlforblog[:60b] start element
-- ::32.922 xmlforblog[:60b] found characters
-- ::32.922 xmlforblog[:60b] start element
-- ::32.923 xmlforblog[:60b] found characters
-- ::32.923 xmlforblog[:60b] end element
-- ::32.923 xmlforblog[:60b] found characters
-- ::32.923 xmlforblog[:60b] start element
-- ::32.924 xmlforblog[:60b] found characters
-- ::32.924 xmlforblog[:60b] end element
-- ::32.924 xmlforblog[:60b] found characters
-- ::32.924 xmlforblog[:60b] start element
-- ::32.925 xmlforblog[:60b] found characters
-- ::32.925 xmlforblog[:60b] end element
-- ::32.925 xmlforblog[:60b] found characters
-- ::32.925 xmlforblog[:60b] end element
-- ::32.926 xmlforblog[:60b] found characters
-- ::32.926 xmlforblog[:60b] start element
-- ::32.928 xmlforblog[:60b] found characters
-- ::32.929 xmlforblog[:60b] start element
-- ::32.929 xmlforblog[:60b] found characters
-- ::32.929 xmlforblog[:60b] end element
-- ::32.930 xmlforblog[:60b] found characters
-- ::32.930 xmlforblog[:60b] start element
-- ::32.930 xmlforblog[:60b] found characters
-- ::32.930 xmlforblog[:60b] end element
-- ::32.931 xmlforblog[:60b] found characters
-- ::32.931 xmlforblog[:60b] start element
-- ::32.931 xmlforblog[:60b] found characters
-- ::32.931 xmlforblog[:60b] end element
-- ::32.931 xmlforblog[:60b] found characters
-- ::32.932 xmlforblog[:60b] end element
-- ::32.932 xmlforblog[:60b] found characters
-- ::32.932 xmlforblog[:60b] start element
-- ::32.932 xmlforblog[:60b] found characters
-- ::32.933 xmlforblog[:60b] start element
-- ::32.933 xmlforblog[:60b] found characters
-- ::32.933 xmlforblog[:60b] end element
-- ::32.933 xmlforblog[:60b] found characters
-- ::32.934 xmlforblog[:60b] start element
-- ::32.934 xmlforblog[:60b] found characters
-- ::32.934 xmlforblog[:60b] end element
-- ::32.934 xmlforblog[:60b] found characters
-- ::32.935 xmlforblog[:60b] start element
-- ::32.935 xmlforblog[:60b] found characters
-- ::32.935 xmlforblog[:60b] end element
-- ::32.935 xmlforblog[:60b] found characters
-- ::32.936 xmlforblog[:60b] end element
-- ::32.936 xmlforblog[:60b] found characters
-- ::32.936 xmlforblog[:60b] end element
-- ::32.936 xmlforblog[:60b] parse end
-- ::32.936 xmlforblog[:60b] (
{
age = ;
firstName = Wythe;
id = ;
lastName = xu;
},
{
age = ;
firstName = li;
id = ;
lastName = si;
},
{
age = ;
firstName = Dipen;
id = ;
lastName = Shah;
}
)
执行结果
而我们的处理主要是在 开始标签、找到字符串 (
-(void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict
-(void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string
) 中。
遇到开始标签时,我们现判断标签,名字,如果是person,表明接下来就是person的信息,这样我们就先创建一个可变字典,以便将来存放它的值。
-(void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict
{
_currenttag = elementName; if ([_currenttag isEqualToString:@"person"]) {
NSString *_id = [attributeDict objectForKey:@"id"];
NSMutableDictionary *dict = [[NSMutableDictionary alloc]init];
[dict setObject:_id forKey:@"id"];
[_person addObject:dict];
} NSLog(@"start element");
}
在找到字符串时,我们就是通过判断当前标签名,将对应的信息保存到刚刚创建的字典中
-(void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string
{
NSMutableDictionary *dict = [_person lastObject]; if ([_currenttag isEqualToString:@"firstName"] && dict) {
[dict setObject:string forKey:@"firstName"]; }
if ([_currenttag isEqualToString:@"lastName"] && dict) {
[dict setObject:string forKey:@"lastName"];
}
if ([_currenttag isEqualToString:@"age"] && dict) {
[dict setObject:string forKey:@"age"];
} NSLog(@"found characters");
}
不断循环这样的过程,最后我们就可以解析出整个XML文档。
另外说一句,这只是解析一般的文档,如果你跟我曾经一样学会这个就去解析豆瓣API的XML文档,会发现行不通。这时因为许多网站因为它的数据较多,为了避免标签的重复,使用了命名空间,带有命名空间的XML文档解析和这稍有不同。
以后我会写带命名空间的XML文档解析,敬请期待。
拖了快一个月了,今天终于写完。以后不能这么懒了
IOS 解析XML文档的更多相关文章
-
Objective-C ,ios,iphone开发基础:使用GDataXML解析XML文档,(libxml/tree.h not found 错误解决方案)
使用GDataXML解析XML文档 在IOS平台上进行XML文档的解析有很多种方法,在SDK里面有自带的解析方法,但是大多情况下都倾向于用第三方的库,原因是解析效率更高.使用上更方便 这里主要介绍一下 ...
-
网络电视精灵~分析~~~~~~简单工厂模式,继承和多态,解析XML文档,视频项目
小总结: 所用技术: 01.C/S架构,数据存储在XML文件中 02.简单工厂模式 03.继承和多态 04.解析XML文档技术 05.深入剖析内存中数据的走向 06.TreeView控件的使用 核心: ...
-
使用dom4j解析XML文档
dom4j的包开源包,不属于JDK里面,在myeclipse中要单独导入在项目中,这里不累赘了 做这个过程,很慢,因为很多方法没用过不熟悉,自己得去查帮助文档,而且还得去试,因为没有中文版,英文翻译不 ...
-
四种生成和解析XML文档的方法详解(介绍+优缺点比较+示例)
众所周知,现在解析XML的方法越来越多,但主流的方法也就四种,即:DOM.SAX.JDOM和DOM4J 下面首先给出这四种方法的jar包下载地址 DOM:在现在的Java JDK里都自带了,在xml- ...
-
java解析xml文档(dom)
DOM解析XML文档 读取本地的xml文件,通过DOM进行解析,DOM解析的特点就是把整个xml文件装载入内存中,形成一颗DOM树形结构,树结构是方便遍历和和操纵. DOM解析的特性就是读取xml文件 ...
-
java 解析XML文档
Java 解析XML文档 一.解析XML文档方式: 1.DOM方式:将整个XML文档读取到内存中,按照XML文件的树状结构图进行解析. 2.SAX方式:基于事件的解析,只需要加载XML中的部分数据,优 ...
-
DOM生成XML文档与解析XML文档(JUNIT测试)
package cn.liuning.test; import java.io.File; import java.io.IOException; import javax.xml.parsers.D ...
-
[置顶] stax解析xml文档的6种方式
原文链接:http://blog.csdn.net/u011593278/article/details/9745271 stax解析xml文档的方式: 基于光标的查询: 基于迭代模型的查找: 基于过 ...
-
浅谈用java解析xml文档(四)
继续接上一文,这一阵子因为公司项目加紧,导致最后一个解析xml文档的方式,还没有总结,下面总结使用dom4J解析xml. DOM4J(Document Object Model for Java) 使 ...
随机推荐
-
git push 使用总结
git push命令用于将本地分支的更新,推送到远程主机.它的格式与git pull命令相仿. $ git push <远程主机名> <本地分支名>:<远程分支名> ...
-
我没发现Mvc里的 web.config 有什么用。
实验过程 由于 Mvc2+ 引入 Area ,导致文件夹结构发生变化. Mvc下的 web.config 所在的位置是: ~/Areas/MySystem/Views/Web.config 对应的请求 ...
-
RouterOS软路由设置固定IP+PPPOE
内网: IP:192.168.10.254/24 网关:192.168.10.254 外网: IP:218.17.172.17/28 子网掩码:255.255.255.240 网关:218.17.17 ...
-
Jquery类级别与对象级别插件开发
jQuery插件的开发包括两种: 一种是类级别的插件开发,即给jQuery添加新的全局函数,相当于给jQuery类本身添加方法.jQuery的全局函数就是属于jQuery命名空间的函数,另一种是对象级 ...
-
linux常用命令--diff
diff是Unix系统的一个很重要的工具程序. 它用来比较两个文本文件的差异,是代码版本管理的基石之一.你在命令行下,输入: $ diff <变动前的文件> <变动后的文件> ...
-
Android底部导航栏——FrameLayout + RadioGroup
原创文章,转载请注明出处http://www.cnblogs.com/baipengzhan/p/6285881.html Android底部导航栏有多种实现方式,本文详细介绍FrameLayout ...
-
POJ 3347 Kadj Squares (计算几何)
题目: Description In this problem, you are given a sequence S1, S2, ..., Sn of squares of different si ...
-
【Mybatis】mybatis使用示例
BusinessAnalysisMapper.java import com.chinamobile.epic.dao.model.PerformanceMetricAnalysis; import ...
-
Shiro系列(2) - 权限模型以及权限分配的两种方式
1. *账户分配权限用户需要被分配相应的权限才可访问相应的资源.权限是对于资源的操作一张许可证.给用户分配资源权限需要将权限的相关信息保存到数据库.这些相关内容包含:用户信息.权限管理.用户分配的权 ...
-
python的pip升级问题
近来由于pip升级为10.0.1了,导致使用pip命令报错,使用过很多方法,最终找到一种相对靠谱的方法,一下是步骤: 进入https://pypi.python.org/pypi/pip 下载pip- ...