数据解析

在网络开发中,从服务器获取的二进制数据包括:

> html

>图片

>视频

>音频

> zip等

除了以上文件格式之外,最常见的数据莫过于JSON,偶尔也会有XML。

无论JSON还是XML都是一种特殊格式的字符串,按照特定的规则描述数据结构。

一、准备工作--->安装Apache服务器

>参考备课笔记/笔记/02-Apache服务器安装笔记.m

二、JSON

1、JSON格式介绍

# JSON本质上,就是一个“特殊格式”的字符串。

> JSON是网络上用来传输数据使用最广泛的数据格式,没有之一。

> JSON出生草根,是Javascript的子集,是Javascript的SON,专门用来描述数据结构。

> Javascrpit是做网页开发使用的一种"脚本"语言

> Javascrpit & Java没有任何关系!就好像雷锋和雷峰塔之间的关系。

#参考网站:http//www.w3cschool.cc。w3c是国际互联网组织的缩写。

2、JSON的语法规则

>数据以key/value键值对表示

>数据由逗号分割

>花括号保存对象,对应OC的NSDictionary。

>方括号保存数组,对应OC的NSArray。

3> JSON值

*数字(整数或浮点数)

*字符串(在双引号中)

*逻辑值(true或false)

*数组(在方括号中)

* null

4、JSON解析--->解析天气预报

>序列化:在向服务器发送数据之前,将NSArray/NSDcitionary转成二进制的过程。

>反序列化:在从服务器接收到数据之后,将二进制数据转换成NSArray/NSDcitionary的过程。

/**

*解析天气预报

*/

- (void)loadData{

//创建url

NSURL *url = [NSURL URLWithString:@"http://www.weather.com.cn/adat/sk/101010100.html"];

//创建请求对象

NSURLRequest *request = [NSURLRequest requestWithURL:url cachePolicy:0timeoutInterval:10.0];

//发送异步请求

[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {

NSLog(@"data = %@",data);

/*

>序列化:在向服务器发送数据之前,将NSArray/NSDcitionary转成二进制的过程。

>反序列化:在从服务器接收到数据之后,将二进制数据转换成NSArray/NSDcitionary的过程。

*/

NSLog(@"%@",[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]);

/*

NSJSONReadingMutableContainers = (1UL << 0),容器可变

NSJSONReadingMutableLeaves = (1UL << 1),叶子可变

NSJSONReadingAllowFragments(碎片) = (1UL << 2)顶级节点可以既不是数组也不是字典

*/

NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:0error:NULL];

NSLog(@"%@市温度%@风向%@风力%@ ",dict[@"weatherinfo"][@"city"],dict[@"weatherinfo"][@"temp"],dict[@"weatherinfo"][@"WD"],dict[@"weatherinfo"][@"WS"]);

}];

}

5、JSON解析第三方框架

>常见的JSON解析第三方框架

* JSONKit(最快)

* SBJson

* TouchJSON

#以上三个框架的性能依次降低。

>介绍JSONKit第三方框架的目的

* JSON的解析并不是表面上这么简单

*官方说JSONKit比苹果原生的JSON解析速度要快。

* JSONKit在很多老的项目中仍然在使用

#知道JSONKit说明我们是资深的iOS程序员。

* JSONKit已经在2012年停止更新,适用于iOS5.0之前的版本开发使用。(苹果原生的NSJSONSerialization是iOS5.0之后出现)

*了解ARC & MRC混编的方法

> JSONKit解析和性能测试

使用步骤:

*下载框架:http://github.com/johnezang/JSONKit

*导入框架文件:JSONKit.h & JSONKit.m

*设置MRC标记

>选择"项目"-"Build Phases"-"Compile Sources"

>找到JSONKit.m并且在Compiler Flags中添加-fno-objc-arc。

# -fno-objc-arc告诉编译器,编译JSONKit.m时不使用ARC。

*修改错误

>利用自动修复功能,修改两处isa的错误。

*反序列化

>idresult =  [[JSONDecoder decoder] objectWithData:data];

使用代码和测试如下:

intlargeNumber =100*1000;

- (void)loadData{

//创建url

NSURL *url = [NSURL URLWithString:@"http://localhost/demo.json"];

//创建请求对象

NSURLRequest *request = [NSURLRequest requestWithURL:url cachePolicy:0timeoutInterval:10.0];

//发送异步请求

[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {

/*

官方说JSONKit比苹果原生的JSON解析速度要快。我们可以通过测试验证下

JSONKit:4.907761秒

官方:0.299994秒

通过测试得出:苹果原生的JSON解析速度远远比JSONKit快。

如果工作中遇到使用第三方框架解析JSON的旧项目,建议替换为苹果原生的。

替换的原则:看项目是否要支持iOS5.0之前的版本。如果不需要果断替换。

替换步骤:

1>备份(给自己一个后悔的机会)

2>删除JSONKit.h & .m文件

3>哪里出错改哪里

*/

CFAbsoluteTime start = CFAbsoluteTimeGetCurrent();

for(intindex =0; index < largeNumber; index ++) {

//            NSMutableDictionary *result =  [[JSONDecoder decoder] objectWithData:data];

NSMutableDictionary *result = [NSJSONSerialization JSONObjectWithData:data options:0error:NULL];

}

NSLog(@"-----endTime = %f",CFAbsoluteTimeGetCurrent() - start);

//JKDictionary其实就是NSMutableDictionary的子类

//         NSLog(@"------%@,%@",result,result[@"message"]);

}];

}

6、PList解析

> PList主要在苹果开发中常用,很多后台并不会返回Plist的格式数据。有关PList的反序列化知道即可。

>代码如下:

/**

*解析plist数据

*/

- (void)loadData{

//创建url

NSURL *url = [NSURL URLWithString:@"http://localhost/videos.plist"];

//创建请求对象

NSURLRequest *request = [NSURLRequest requestWithURL:url cachePolicy:0timeoutInterval:10.0];

//发送异步请求

[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {

/**

参数:

- data:要反序列化的二进制数据

- options:选项,位移枚举类型

NSPropertyListImmutable = 0不可变,

NSPropertyListMutableContainers = 1容器可变,

NSPropertyListMutableContainersAndLeaves = 2容器和叶子可变

- format:如果不希望知道格式,传人NULL即可

- error:错误信息

*/

idresult = [NSPropertyListSerialization propertyListWithData:data options:0format:NULLerror:nil];

NSLog(@"%@   %@",result,[result class]);

}];

}

三、XML解析

1、XML介绍

> XML的英文全称(eXtensible Markup Language),中文名称可扩展标记语言。

> XML的特点:语法规则很简单,且很有逻辑。

#出身名门,w3c制定,微软和IBM曾经共同大力推荐过的数据格式。

>专门设计用来传输和存储数据

>与HTML的区别:HTML的标记不是所有的都需要成对出现,

XML要求所有的标记必须成对出现。

HTML标记不区分大小写,它则大小敏感,即区分大小写。

2、XML解析的方式

> DOM解析

*是在MAC使用的解析方式。

*内存消耗极大,不适用于手机。因此苹果没有提供手机上DOM解析。

* iPhone无法直接使用DOM方式解析XML。

> SAX解析

*是只读的方式,从上向下的方式解析

*是苹果提供的解析方式

*使用NSXMLParser通过代理实现解析。

3、SAX解析步骤

1>开始文档--准备工作

2>开始"节点"

3>发现节点内部的内容,每一个节点,可能需要多次才能找完

4>结束"节点"

5>结束文档--解析结束

#以上步骤,2,3,4会不断循环,一直到所有解析完成。

4、代码演示XML的解析。

/**

*加载xml数据

*/

- (void)loadData{

//创建url

NSURL *url = [NSURL URLWithString:@"http://localhost/videos.xml"];

//创建请求对象

NSURLRequest *request = [NSURLRequest requestWithURL:url cachePolicy:0timeoutInterval:10.0];

//发送异步请求(由于解析xml是一个耗时的过程,所以新开启队列)

[NSURLConnection sendAsynchronousRequest:request queue:[[NSOperationQueue alloc] init] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {

// 1.创建xml解析器

NSXMLParser *parser = [[NSXMLParser alloc] initWithData:data];

// 2.设置代理来解析

parser.delegate =self;

// 3.解析器开始解析--一旦开始解析,后续的解析工作全部交给代理完成

[parser parse];

}];

}

>通过NSLog打印,确认XML解析思路

#pragma mark  - NSXMLParserDelegate代理方法

/**

*开始解析

*/

- (void)parserDidStartDocument:(NSXMLParser *)parser {

NSLog(@"1、开始解析");

}

/**

*解析到一个开始节点调用

*/

- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict {

NSLog(@"2、开始节点%@ %@",elementName,attributeDict);

}

/**

*解析到节点文字调用一个节点的内容可以会调用多次该方法

*/

- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string {

NSLog(@"3、======> %@",string);

}

/**

*解析到结束节点调用elementName没有反斜线/

*/

- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName {

NSLog(@"4、结束节点------%@",elementName);

}

/**

*解析结束调用

*/

- (void)parserDidEndDocument:(NSXMLParser *)parser{

NSLog(@"5、结束文档");

}

5、画XML解析思维导图

6、XML解析代码实现

#pragma mark  - NSXMLParserDelegate代理方法

/**

*开始解析

*/

- (void)parserDidStartDocument:(NSXMLParser *)parser {

//清空数组

[self.videos removeAllObjects];

}

/**

*解析到一个开始节点调用

*/

- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict {

if([elementName isEqualToString:@"video"]) {

//创建video模型

self.currentVideo = [[Video alloc] init];

//设置id

self.currentVideo.videoId =@([attributeDict[@"videoId"] integerValue]);

}

//清空当前正在拼接的字符内容

[self.elementString setString:@""];

//错误写法

//self.elementString = @"";

}

/**

*解析到节点文字调用一个节点的内容可以会调用多次该方法

*/

- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string {

[self.elementString appendString:string];

}

/**

*解析到结束节点调用elementName没有反斜线/

*/

- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName {

//如果elementName是video,则添加到数组中

/*最笨方法

if ([elementName isEqualToString:@"video"]){

[self.videos addObject:self.currentVideo];

} else if ([elementName isEqualToString:@"name"]){

self.currentVideo.name= self.elementString;

} else if ([elementName isEqualToString:@"length"]){

self.currentVideo.length = @(self.elementString.integerValue);

} else if ([elementName isEqualToString:@"desc"]){

self.currentVideo.desc = self.elementString;

} else if ([elementName isEqualToString:@"imageURL"]) {

self.currentVideo.imageURL = self.elementString;

} else if ([elementName isEqualToString:@"videoURL"]){

self.currentVideo.videoURL = self.elementString;

} else if ([elementName isEqualToString:@"teacher"]){

self.currentVideo.teacher = self.elementString;

}

*/

//如果是name,length,desc....则设置属性

//简便方法

if([elementName isEqualToString:@"video"]){

[self.videos addObject:self.currentVideo];

}elseif(![elementName isEqualToString:@"videos"]){

//使用KVC设置属性, KVC是一种以字符串形式间接设置数据的方式

[self.currentVideo setValue:self.elementString forKey:elementName];

}

}

/**

*解析结束调用

*/

- (void)parserDidEndDocument:(NSXMLParser *)parser{

NSLog(@"%@",self.videos);

}

四、tableView展示数据

1、展示视频数据--Storyboard自定义cell

2、设置时间格式

> HMVideo模型中提供属性

//视频时长

@property(nonatomic,strong,readonly) NSString *time;

#计算时间方式有两种

#方式一:重写time属性getter方法

- (NSString *)time {

intlen =self.length.intValue;

return[NSString stringWithFormat:@"%02d:%02d:%02d", len /3600, (len %3600) /60, (len %60)];

}

#方式二:

- (void)setLength:(NSNumber *)length {

_length = length;

intlen =self.length.intValue;

_time = [NSString stringWithFormat:@"%02d:%02d:%02d", len /3600, (len %3600) /60, (len %60)];

}

3、SDWebImage下载头像

UIImage *placeholder = [UIImage imageNamed:@"user_default"];

// SDWebImageLowPriority:会在表格滚动的时候,暂时图像下载,性能会好很多

// SDWebImageRetryFailed:图片下载失败后重试

[self.iconView sd_setImageWithURL:video.fullImageUrl placeholderImage:placeholder options:SDWebImageRetryFailed | SDWebImageLowPriority];

4、集成下列刷新控件-->通过代码和storyboard

/**

*集成下拉刷新控件(通过代码)

*/

- (void)setRefreshControl{

//集成下拉控件(不需要设置frame)

self.refreshControl = [[UIRefreshControl alloc] init];

//监听下拉事件

[self.refreshControl addTarget:selfaction:@selector(loadData) forControlEvents:UIControlEventValueChanged];

//设置提示文字,颜色

NSMutableAttributedString *attributedTitle = [[NSMutableAttributedString alloc] initWithString:@"努力加载中……"attributes:@{NSFontAttributeName:[UIFont systemFontOfSize:15],NSForegroundColorAttributeName:[UIColor redColor]}];

self.refreshControl.attributedTitle = attributedTitle;

//设置菊花颜色

self.refreshControl.tintColor = [UIColor grayColor];

}

# UIRefreshControl使用说明

1>目前只对UITableviewController有用

2>只能下拉刷新,不能上拉刷新

3> init或者viewdidload中创建UIRefreshControl,设置文字,颜色等信息

4>系统自动管理UIRefreshControl,自动添加到tableview视图中;

5>给UIRefreshControl添加方法,当值改变的时候调用,方法用于数据请求

6>该方法中请求数据确认完成之后,调用endRefreshing方法,关闭刷新

五、XML解析重构

1>新建一个模型类HMSAXVideo,专门做解析工作。

2>提供类方法:+(void)saxParser:(NSData *)data finished:(void(^)(NSArray *))finished。

//如果回调的block在当前方法不执行,需要用一个属性记录

+(void)saxParser:(NSData *)data finished:(void(^)(NSArray *))finished{

//使用断言确保外界传人的回调不为空,如果为空,则崩溃提示

NSAssert(finished !=nil,@"必须传人回调");

// 0.创建解析模型

HMSAXVideo *sax = [[HMSAXVideo alloc] init];

//记录回调block

sax.finishedBlock = finished;

// 1.创建xml解析器

NSXMLParser *parser = [[NSXMLParser alloc] initWithData:data];

// 2.设置代理来解析

parser.delegate = sax;

// 3.解析器开始解析

// parse:是同步方法,会等所用代理方法执行完毕后,才会执行后面的代码

[parser parse];

#warning不能在parse方法调用后记录回调

//记录回调block

// sax.finishedBlock = finished;

}

六、Copy,NSNumber,id

1、为什么模型HMVideo中的属性都使用copy?改为strong运行看看效果。

> strong:在设置数值的时候,仅仅是引用计算+1

> copy:在设置数值的时候,如果有方法是可变的,会copy一个不可变的副本。copy动作是在setter方法中执行的。

>画图解析使用strong导致的效果。

>如果属性使用strong:

#下面的代码

[self.currentVideo setValue:self.elementString forKey:elementName];

#修改为

[self.currentVideo setValue:self.elementString.copy forKey:elementName];

>重新了属性的setter方法,就是接管了原有系统提供的setter方法。

#如果重新了copy属性的setter方法,一定要copy,否则外面的copy属性就是strong属性。

比如重写了模型的name属性的setter方法:

- (void)setName:(NSString *)name {

_name = [name copy];

//    _name = name;属性就相当于strong了

}

你可能感兴趣的:(数据解析)