03-IOSCore - XML及解析、Plist
一、XML 可扩展标记语言
是什么?是一段有规范的字符串,
用在哪?用在任何地方
语法:
* 结点Node
<结点名 属性名="属性值">
结点内容
</结点名>
* 结点的开始和结尾配对
* 结点内可以有子结点
* 结点内可以有文本
* 结点名和属性名区分大小写
* 结点不能交叉
* 结点名和属性名 随便写
术语:
结点、属性
叶子结点 :没有子结点的结点
<a b="c"></a>可以简化<a b="c" />
非叶子结点 :有子结点的结点
对于存储数据的xml,非叶子结点几乎不存储任何文本,对于一些xml的变体xhtml,就会出现大量的非叶子子结点存储文本的现象,比如:
<div>
<h1>...</h1>
<p>ggg...<a>...</a></p>
</div>
根结点 :没有父结点的结点,一个xml只能有一个根结点
层级关系:
子结点
父结点
兄弟结点
其他:特殊符号不能直接写在xml中,比如<要写成< >>
xml解析:
XML文件结构解析
1 事件驱动 SAX iOS采用事件驱动解析
ios具体解析xml文档步骤:
1) 准备好xml文档的路径
2) 转换为data数据
3) 创建NSXMLParser对象调用initWithData把data传进来
4) 设置delegate,然后发送parse消息解析文档
NSString *path = @"/.../message.xml";
NSData *data = [NSDatadataWithContentsOfFile:path];
NSXMLParser *xmlParser = [[NSXMLParseralloc] initWithData:data];
xmlParser.delegate = self;
[xmlParser parse];
delegate中几个重要的API:
// 开始解析xml文档
-(void)parserDidStartDocument:(NSXMLParser *)parser{
NSLog(@"开始解析...");
}
// 发现结点节点名 + 属性
-(void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict{
NSLog(@"开始节点名:%@",elementName);
[attributeDict enumerateKeysAndObjectsUsingBlock:^(id key, id obj, BOOL *stop) {
NSLog(@"属性名:%@, 属性值:%@",key,obj);
}];
}
// 结束结点
-(void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName{
NSLog(@"结束结点:%@",elementName);
}
// 发现文本
-(void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string{
NSLog(@"发现文本:%@",string);
}
// 结束解析xml文档
-(void)parserDidEndDocument:(NSXMLParser *)parser{
NSLog(@"结束文档解析");
}
实现xml解析步骤:
/*
解析xml文档:
根据本例中message.xml文件解析
如果发现Message节点就创建一个类,同时根据attributeDict可以获取到属性order并赋值
为了可以获取xml文件中文本的值并赋给message类的属性(节点名),需要记下最近的那个节点名lastElementName
在发现文本方法中判断lastElementName是否和message类中的属性相同,如果相同就赋值
注意:文本直接赋值会出问题,"\n"这个会覆盖赋值好的文本,所以在结束结点中应该给lastElementName置空
快速修改本文件中所有的属性名技巧:
选中属性名后右击选中refactor -> rename
*/
#import "MXMessageParse.h"
#import "MXMessage.h"
@interfaceMXMessageParse ()
@property(nonatomic,strong) MXMessage *lastMessage;
@property(nonatomic,copy) NSString *lastElementName;
@property(nonatomic,strong) NSMutableArray *messages;
@end
@implementation MXMessageParse
-(void)parser{
NSString *path = @"/Users/tarena/yz/第三阶段(UI核心_Model赵哲)/day03/day0302_xml_parse解析/message.xml";
NSData *data = [NSDatadataWithContentsOfFile:path];
NSXMLParser *xmlParser = [[NSXMLParseralloc] initWithData:data];
xmlParser.delegate = self;
[xmlParser parse];
}
// 开始解析xml文档
-(void)parserDidStartDocument:(NSXMLParser *)parser{
NSLog(@"开始解析...");
}
// 发现结点节点名 + 属性
-(void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict{
NSLog(@"开始节点名:%@",elementName);
[attributeDict enumerateKeysAndObjectsUsingBlock:^(id key, id obj, BOOL *stop) {
NSLog(@"属性名:%@, 属性值:%@",key,obj);
}];
if ([elementName isEqualToString:@"Message"]) {
self.lastMessage = [[MXMessagealloc] init];
self.lastMessage.order = [attributeDict[@"order"] intValue];
}elseif ([elementName isEqualToString:@"messages"]){
self.messages = [NSMutableArrayarray];
}
self.lastElementName = elementName;
}
// 结束结点
-(void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName{
NSLog(@"结束结点:%@",elementName);
if ([elementName isEqualToString:@"Message"]) {
[self.messagesaddObject:self.lastMessage];
}
self.lastElementName = Nil;
}
// 发现文本
-(void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string{
NSLog(@"发现文本:%@",string);
if ([self.lastElementNameisEqualToString:@"fromMe"]) {
self.lastMessage.fromMe = [string isEqualToString:@"YES"];
}elseif ([self.lastElementNameisEqualToString:@"type"]){
NSString *type = string;
if ([type isEqualToString:@"TXT"]) {
self.lastMessage.type = MXMessageTypeText;
}
}elseif ([self.lastElementNameisEqualToString:@"content"]){
self.lastMessage.content = string;
}
}
// 结束解析xml文档
-(void)parserDidEndDocument:(NSXMLParser *)parser{
NSLog(@"结束文档解析");
}
2 树状模型 DOM
XML结构 <-> 对象
Node *rootNode = [parser parse:path];
NSString *elementName = rootNode.name;
Node *messageNode = [rootNode childNodeOfName@"Message"];
messageNode.name;
[messageNode attributeByName:@"order"];
3 优缺点对比
事件驱动 树状模型
使用困难 使用简单
边读取边解析 读取完整后在解析
摘取部分重要信息 从完整信息中搜索
二、plist (Property List)
是一个有固定格式的xml文件,用来存储中小型数据,在ios中使用
plist可以存储的数据类型:
NSArray
NSDictionary
根节点必须是上面两个类型之一
NSString
NSNumber
NSData
Boollean
plist解析:
NSArray arrayWithContentsOfFile
NSDictionary dictionaryWithContentsOfFile
plist的解析过程是
plist(XML) <--> OC对象结构 互相转换
对应关系:
OC
<Message order="">
+ fromMe
+ content
MXL
Message
<formMe></formMe>
<content></content>
Plist
formMe ->
content ->
读取plist中的数据
- (void)viewDidLoad
{
[superviewDidLoad];
// 读取plist数据
NSMutableArray *arr = [NSMutableArrayarray];
NSString *path = [[NSBundlemainBundle] pathForResource:@"messages"ofType:@"plist"];
NSArray *array = [NSArrayarrayWithContentsOfFile:path];
for (NSDictionary *dic in array) {
MXMessage *message = [[MXMessagealloc] init];
message.content = dic[@"content"];
message.fromMe = [dic[@"fromMe"] boolValue];
if ([dic[@"type"] isEqualToString:@"TXT"]) {
message.type = MXMessageTypeText;
}
[arr addObject:message];
}
for (MXMessage *message in arr) {
NSLog(@"%@",message.content);
}
}
写入:
writeToFile:
必须确保字典/数组内的所有对象均是plist支持的对象
复杂对象树状结构用plist表达
area
代码如下:
- (void)viewDidLoad
{
[superviewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
NSLog(@"%@",NSHomeDirectory());
NSString *documentsPath = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES)[0];
NSString *plistPath = [documentsPath stringByAppendingPathComponent:@"china.plist"];
[[NSFileManagerdefaultManager] createFileAtPath:plistPath contents:Nilattributes:Nil];
// 对象 -> 字典
TRArea * puDong = [[TRAreaalloc] init];
puDong.areaName = @"浦东";
NSDictionary *dicPuDong = @{@"areaName": puDong.areaName};
TRArea * shangHai = [[TRAreaalloc] init];
shangHai.areaName = @"上海";
shangHai.subAreas = @[dicPuDong];
NSDictionary *dicShangHai = @{@"areaName": shangHai.areaName,
@"subAreas": shangHai.subAreas};
TRArea * china = [[TRAreaalloc] init];
china.areaName = @"中国";
china.subAreas = @[dicShangHai];
NSDictionary *dicChina = @{@"areaName": china.areaName,
@"subAreas": china.subAreas};
[dicChina writeToFile:plistPath atomically:YES];
}
作业
1. TMessage plist存储支持
2. TContact plist存储支持(放document下)
1) 定义一个plist文件格式
2) 写解析代码
3) 实现读取文件内容显示消息/联系人信息
4) 支持保存
发新消息/建新对象
把messages/contacts数组 转化成
plist支持的数组(数组结构 还要跟定义的结构一样)
03-IOSCore - XML及解析、Plist的更多相关文章
- 玩转iOS开发 - JSON 和 Xml 数据解析
前言 Json 和xml是网络开发中经常使用的数据格式,JSON轻量级.xml相对较复杂.所以如今用JSON的比例很大.基本上从server获取的返回数据都是JSON格式的,作为iOS开发人员,解析J ...
- JavaEE XML DOM解析之DOM4J
DOM解析之DOM4J @author ixenos DOM4J常用API 读取xml文档: Document doc = new SAXReader().read("xml文件" ...
- iOS 应用数据存储方式(XML属性列表-plist)
iOS 应用数据存储方式(XML属性列表-plist) 一.ios应用常用的数据存储方式 1.plist(XML属性列表归档) 2.偏好设置 3.NSKeydeArchiver归档(存储自定义对象) ...
- 通过正则表达式实现简单xml文件解析
这是我通过正则表达式实现的xml文件解析工具,有些XHTML文件中包含特殊符号,暂时还无法正常使用. 设计思路:常见的xml文件都是单根树结构,工具的目的是通过递归的方式将整个文档树装载进一个Node ...
- JAVA 中XML的解析
XML: 可扩展标记语言(extensible Markup Language) 用于标记电子文件使其具有结构性的标记语言.XML可以用来标记数据.定义数据类型,是一种允许用户对自己的标记语言进行定 ...
- xml的解析与创建——bing到youdao导入文件的转换
首先是为了解决一个问题:如何将必应单词本中记录的单词转入到有道词典中去.实际上,必应词典可以导出xml文件,但是该文件有道词典无法解析.这里涉及到xml的解析和创建了. 代码如下: import ja ...
- iOS-数据持久化基础-JSON与XML数据解析
解析的基本概念 所谓“解析”:从事先规定好的格式串中提取数据 解析的前提:提前约定好格式.数据提供方按照格式提供数据.数据获取方按照格式获取数据 iOS开发常见的解析:XML解析.JSON解析 一.X ...
- 八、Android学习第七天——XML文件解析方法(转)
(转自:http://wenku.baidu.com/view/af39b3164431b90d6c85c72f.html) 八.Android学习第七天——XML文件解析方法 XML文件:exten ...
- iOS开发UI篇—ios应用数据存储方式(XML属性列表-plist)
iOS开发UI篇—ios应用数据存储方式(XML属性列表-plist) 一.ios应用常用的数据存储方式 1.plist(XML属性列表归档) 2.偏好设置 3.NSKeydeArchiver归档(存 ...
随机推荐
- aop切入点表达式
1.切入点表达式:对指定的方法进行拦截,并且生成代理表达式. 2.拦截所有public方法 <aop:pointcut expression="execution(public * * ...
- 「Foundation」集合
一.NSArray和NSMutableArray (一)NSArray不可变数组 (1)NSArray的基本介绍 NSArray是OC中使用的数组,是面向对象的,以面向对象的形式操纵对象,是不可变数组 ...
- tomcat手动发布
有些时候不能通过eclipse中的server服务发布工程,这时候就只能通过手动配置进行发布了 如下: 配置发布路径:D:\Program Files\apache-tomcat-6.0.10\con ...
- 通过代码...CGRectmake 创建的控件如何自适应屏幕。
[[UIScreen mainScreen] bounds].size.width [[UIScreen mainScreen] bounds].size.height 得到屏幕的宽高,然后将变量写入 ...
- linux 进程通信
IPC: 管道,FIFO,信号,消息队列(system v/ posix),共享内存(system v/ posix),socket 同步机制: 互斥锁,条件变量,记录上锁, 信号量(system ...
- 数据切分——Mysql分区表的管理与维护
关于Mysql分区表的介绍可以参考: http://blog.csdn.net/jhq0113/article/details/44592865 关于Mysql分区表的创建可以参考: http://b ...
- 「操作系统」: Conditional Move Instructions(trap)
Not all conditional expressions can be compiled using conditional moves. Most significantly, the abs ...
- 1104--DNA排序
问题描述: 逆序数可以用来描述一个序列混乱程度的量.例如,“DAABEC”的逆序数为5,其中D大于它右边的4个数·,E大于它右边的1的个数,4+1=5,又如,“ZWQM”的逆序数为3+2+1+0=6. ...
- Java程序如何自动在线升级
有时候我们的程序需要连接服务器检测新版本,如果发现新版本则需要自动下载升级.这种需求在Linux下还好说,但在windows下如何替换正在运行的程序文件呢? 当然有办法,步骤如下: 1. 将我们的程序 ...
- 如何成为CSDN博客专家
先看一下官方给出的要求: 申请CSDN博客专家应具备的条件: 1.原创IT类文章总数超过20篇,并且最近一个月内发布了新的原创IT类文章. 2.博客文章总的浏览量超过5万次以上. 3.文章内容的质量很 ...