iOS开发过程中,网络数据的传输过程一般是:客户端发送请求给服务器,服务器接收到客户端发送的网络请求后返回相应的数据。此时客户端需要把服务器返回的数据转化为前段和移动端开发中使用的数据格式(如OC/java)。后台服务器一般使用php、java、.net进行开发,而前段和移动端使用的一般是OC/JAVA/HTML/CSS/JS,做好前后端的数据交互极为重要,如今数据交互常用的就是JSON和XML。下面就iOS开发过程中的JSON解析和XML解析进行简单的说明。
一、JSON解析
JSON是一种轻量级的数据格式,一般用于数据交互。JSON是javascript语言的一个子集.javascript是个脚本语言(不需要编译),用来给HTML增加动态功能(此处的 javascript和java没有半毛钱的关系!)
在iOS中,常见的JSON数据解析方案有4种:
第三方框架: SBJson ,JSONKit, TouchJson.性能从左到右,依次变差.(iOS 5(2011年)以前)
苹果原生(自带):NSJSONSerialization (性能是最好的.iOS5以后推出).
苹果原生的JSON解析工具封装的效果很好,用起来也很方便,下面简要的介绍一下JSON的解析方法。
<1>反序列化(解析):
将从服务器接收到的JSON数据(二进制数据)转换成OC数据类型(NSArray,NSDictionary等.)的过程.
目的: JSON数据 --> OC对象; 得到数据字典或者数据数组
好处: 简化程序的开发,方便后续的字典转模型.
方法:
+(id)JSONObjectWithData:(NSData *)data options:(NSJSONReadingOptions)opt error:(NSError **)error;
<2>序列化:
将数组或者字典发送给服务器之前,转换成二进制数据.
目的:OC对象 --> JSON数据;得到二进制JSON数据 NSData.
好处:方便网络传输,提高传输速度.
方法:
+(NSData *)dataWithJSONObject:(id)obj options:(NSJSONWritingOptions)opt error:(NSError **)error;
此时需要注意在做序列化之前,一定使用 isValidJSONObject 检测一下要序列化的对象,是否能够正确被序列化
还有就是如果后台返回的json数据有误,比如两侧的{}或者[]没有加上,我们要通过代码实现,下面列出实现思路,就不上代码了。
具体思路: 1.发送网络请求 2.用字符串接收JSON数据 3.将JSON格式标准化 4.将标准化之后的JSON字符串转换成二进制数据 5.将标准的JSON二进制数据转化为OC数据
二、解析XML数据
XML 全称是Extensible Markup Language,译作“可扩展标记语言
XML特征:
1、 XML是一种标记语言,很类似HTML
2、XML的设计宗旨是传输数据,而非显示数据
3、XML标签没有被预定义。您需要自行定义标签。
4、XML被设计为具有自我描述性。
5、XML是W3C的推荐标准
XML是独立于软件和硬件的信息传输工具。 目前,XML在Web中起到的作用不会亚于一直作为 Web 基石的 HTML。 XML无所不在。XML是各种应用程序之间进行数据传输的最常用的工具,并且在信息存储和描述领域变得越来越流行。
XML解析三种方式
DOM
通用性强,它会将XML文件的所有内容读取到内存中,然后允许您使用DOM API遍历XML树、检索所需的数据;
简单直观,但需要将文档读取到内存,并不太适合移动设备;
SAX、PULL解析
SAX采用事件驱动,它并不需要解析整个文档;PULL(用于java解析),是Android自带的XML解析器,和SAX基本类似,也是事件驱动,不同的是PULL事件返回的是数值型。
他们的区别为:SAX解析器的工作方式是自动将事件推入注册的事件处理器进行处理,因此你不能控制事件的处理主动结束;而Pull解析器的工作方式为允许你的应用程序代码主动从解析器中获取事件,正因为是主动获取事件,因此可以在满足了需要的条件后不再获取事件,结束解析。这是他们主要的区别。如果在一个XML文档中我们只需要前面一部分数据,但是使用SAX方式或DOM方式会对整个文档进行解析,中间不能终止暂停,尽管XML文档中后面的大部分数据我们其实都不需要解析,因此这样实际上就浪费了处理资源。使用PULL方式正合适。
下面就iOS开发中的XML解析举个例子:
//NSXMLParserDelegate // 1. 实例化 XML 的 SAX 解析器! NSXMLParser *parser = [[NSXMLParser alloc] initWithData:data]; // 2. 设置解析器代理. delegate = self; // 3. 开始解析 XML 文档 // 一旦调用了下面的开始解析方法,就会自动调用代理方法,解析 XML 文档! [parser parse]; #pragma NSXMLParserDelegate - (void)parserDidStartDocument:(NSXMLParser *)parser { NSLog(@"1.XML文档解析开始!"); } // 开始解析元素的时候就会调用!XML文档中有多少个元素就会调用多个次! // elementName:元素名称! // attributeDict:属性字典!当前元素对应的属性字典! - (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(nullable NSString *)namespaceURI qualifiedName:(nullable NSString *)qName attributes:(NSDictionary<NSString *, NSString *> *)attributeDict { NSLog(@"2.开始解析:%@元素,元素属性是:%@",elementName,attributeDict); // 判断:只有元素 vedio 的属性字典是需要的内容! if ([elementName isEqualToString:@"vedio"]) { // 字典转模型 CZVideo *video = [CZVideo videoWithDict:attributeDict]; // 添加到数据源中 [self.videos addObject:video]; } } // 发现元素内容的时候就会调用! // string :元素内容! - (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string { NSLog(@"3.发现元素内容:%@",string); } // 元素解析结束的时候就会调用,XML文档中有多少个元素就会调用多个次! // elementName :元素名称 - (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(nullable NSString *)namespaceURI qualifiedName:(nullable NSString *)qName { NSLog(@"4.元素%@解析结束",elementName); } - (void)parserDidEndDocument:(NSXMLParser *)parser { NSLog(@"5.XML文档解析结束!"); }