博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
IOS中Json解析的四种方法
阅读量:4115 次
发布时间:2019-05-25

本文共 9387 字,大约阅读时间需要 31 分钟。

作为一种轻量级的数据交换格式,json正在逐步取代xml,成为网络数据的通用格式。

有的json代码格式比较混乱,可以使用此“http://www.bejson.com/”网站来进行JSON格式化校验()。此网站不仅可以检测Json代码中的错误,而且可以以视图形式显示json中的数据内容,很是方便。

从IOS5开始,APPLE提供了对json的原生支持(NSJSONSerialization),但是为了兼容以前的ios版本,可以使用第三方库来解析Json。

本文将介绍TouchJson、 SBJson 、JSONKit 和 iOS5所支持的原生的json方法,解析国家气象局API,TouchJson和SBJson需要下载他们的库

TouchJson包下载:

SBJson 包下载:

PS:

国家气象局提供的天气预报接口

接口地址有三个:

http://www.weather.com.cn/data/sk/101010100.html

http://www.weather.com.cn/data/cityinfo/101010100.html

http://m.weather.com.cn/data/101010100.html

第三接口信息较为详细,提供的是6天的天气,关于API所返回的信息请见,全国各城市对应这一个id号,根据改变id好我们就可以解析出来各个城市对应天气;

下面介绍四种方法解析JSON:

首先建立一个新的工程,(注意不要选择ARC机制)添加如下控件:

如上图所示。下面展出程序代码:

文件 ViewController.h 中:

[cpp]
  1. #import <UIKit/UIKit.h>
  2. @interface ViewController : UIViewController
  3. @property (retain, nonatomic) IBOutlet UITextView *txtView;
  4. - (IBAction)btnPressTouchJson:(id)sender;
  5. - (IBAction)btnPressSBJson:(id)sender;
  6. - (IBAction)btnPressIOS5Json:(id)sender;
  7. - (IBAction)btnPressJsonKit:(id)sender;
  8. @end
#import 
@interface ViewController : UIViewController@property (retain, nonatomic) IBOutlet UITextView *txtView;- (IBAction)btnPressTouchJson:(id)sender;- (IBAction)btnPressSBJson:(id)sender;- (IBAction)btnPressIOS5Json:(id)sender;- (IBAction)btnPressJsonKit:(id)sender; @end
文件ViewController.m中主要代码:

(1)使用TouchJSon解析方法:(需导入包:#import "TouchJson/JSON/CJSONDeserializer.h")

[cpp]
  1. //使用TouchJson来解析北京的天气
  2. - (IBAction)btnPressTouchJson:(id)sender {
  3. //获取API接口
  4. NSURL *url = [NSURL URLWithString:@"http://m.weather.com.cn/data/101010100.html"];
  5. //定义一个NSError对象,用于捕获错误信息
  6. NSError *error;
  7. NSString *jsonString = [NSString stringWithContentsOfURL:url encoding:NSUTF8StringEncoding error:&error];
  8. NSLog(@"jsonString--->%@",jsonString);
  9. //将解析得到的内容存放字典中,编码格式为UTF8,防止取值的时候发生乱码
  10. NSDictionary *rootDic = [[CJSONDeserializer deserializer] deserialize:[jsonString dataUsingEncoding:NSUTF8StringEncoding] error:&error];
  11. //因为返回的Json文件有两层,去第二层内容放到字典中去
  12. NSDictionary *weatherInfo = [rootDic objectForKey:@"weatherinfo"];
  13. NSLog(@"weatherInfo--->%@",weatherInfo);
  14. //取值打印
  15. txtView.text = [NSString stringWithFormat:@"今天是 %@ %@ %@ 的天气状况是:%@ %@ ",[weatherInfo objectForKey:@"date_y"],[weatherInfo objectForKey:@"week"],[weatherInfo objectForKey:@"city"], [weatherInfo objectForKey:@"weather1"], [weatherInfo objectForKey:@"temp1"]];
  16. }
//使用TouchJson来解析北京的天气- (IBAction)btnPressTouchJson:(id)sender {    //获取API接口    NSURL *url = [NSURL URLWithString:@"http://m.weather.com.cn/data/101010100.html"];    //定义一个NSError对象,用于捕获错误信息    NSError *error;    NSString *jsonString = [NSString stringWithContentsOfURL:url encoding:NSUTF8StringEncoding error:&error];    NSLog(@"jsonString--->%@",jsonString);    //将解析得到的内容存放字典中,编码格式为UTF8,防止取值的时候发生乱码    NSDictionary *rootDic = [[CJSONDeserializer deserializer] deserialize:[jsonString dataUsingEncoding:NSUTF8StringEncoding] error:&error];    //因为返回的Json文件有两层,去第二层内容放到字典中去    NSDictionary *weatherInfo = [rootDic objectForKey:@"weatherinfo"];    NSLog(@"weatherInfo--->%@",weatherInfo);    //取值打印    txtView.text = [NSString stringWithFormat:@"今天是 %@  %@  %@  的天气状况是:%@  %@ ",[weatherInfo objectForKey:@"date_y"],[weatherInfo objectForKey:@"week"],[weatherInfo objectForKey:@"city"], [weatherInfo objectForKey:@"weather1"], [weatherInfo objectForKey:@"temp1"]];    }
(2)使用SBJson解析方法:(需导入包:#import "SBJson/SBJson.h")

[cpp]
  1. //使用SBJson解析南阳的天气
  2. - (IBAction)btnPressSBJson:(id)sender {
  3. NSURL *url = [NSURL URLWithString:@"http://m.weather.com.cn/data/101180701.html"];
  4. NSError *error = nil;
  5. NSString *jsonString = [NSString stringWithContentsOfURL:url encoding:NSUTF8StringEncoding error:&error];
  6. SBJsonParser *parser = [[SBJsonParser alloc] init];
  7. NSDictionary *rootDic = [parser objectWithString:jsonString error:&error];
  8. NSDictionary *weatherInfo = [rootDic objectForKey:@"weatherinfo"];
  9. txtView.text = [NSString stringWithFormat:@"今天是 %@ %@ %@ 的天气状况是:%@ %@ ",[weatherInfo objectForKey:@"date_y"],[weatherInfo objectForKey:@"week"],[weatherInfo objectForKey:@"city"], [weatherInfo objectForKey:@"weather1"], [weatherInfo objectForKey:@"temp1"]];
  10. }
//使用SBJson解析南阳的天气- (IBAction)btnPressSBJson:(id)sender {    NSURL *url = [NSURL URLWithString:@"http://m.weather.com.cn/data/101180701.html"];    NSError *error = nil;    NSString *jsonString = [NSString stringWithContentsOfURL:url encoding:NSUTF8StringEncoding error:&error];    SBJsonParser *parser = [[SBJsonParser alloc] init];        NSDictionary *rootDic = [parser objectWithString:jsonString error:&error];    NSDictionary *weatherInfo = [rootDic objectForKey:@"weatherinfo"];    txtView.text = [NSString stringWithFormat:@"今天是 %@  %@  %@  的天气状况是:%@  %@ ",[weatherInfo objectForKey:@"date_y"],[weatherInfo objectForKey:@"week"],[weatherInfo objectForKey:@"city"], [weatherInfo objectForKey:@"weather1"], [weatherInfo objectForKey:@"temp1"]];}
(3)使用IOS5自带解析类NSJSONSerialization方法解析:(无需导入包,IOS5支持,低版本IOS不支持)

[cpp]
  1. - (IBAction)btnPressIOS5Json:(id)sender {
  2. NSError *error;
  3. //加载一个NSURL对象
  4. NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://m.weather.com.cn/data/101180601.html"]];
  5. //将请求的url数据放到NSData对象中
  6. NSData *response = [NSURLConnection sendSynchronousRequest:request returningResponse:nil error:nil];
  7. //IOS5自带解析类NSJSONSerialization从response中解析出数据放到字典中
  8. NSDictionary *weatherDic = [NSJSONSerialization JSONObjectWithData:response options:NSJSONReadingMutableLeaves error:&error];
  9. NSDictionary *weatherInfo = [weatherDic objectForKey:@"weatherinfo"];
  10. txtView.text = [NSString stringWithFormat:@"今天是 %@ %@ %@ 的天气状况是:%@ %@ ",[weatherInfo objectForKey:@"date_y"],[weatherInfo objectForKey:@"week"],[weatherInfo objectForKey:@"city"], [weatherInfo objectForKey:@"weather1"], [weatherInfo objectForKey:@"temp1"]];
  11. NSLog(@"weatherInfo字典里面的内容为--》%@", weatherDic );
  12. }
- (IBAction)btnPressIOS5Json:(id)sender {        NSError *error;    //加载一个NSURL对象    NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://m.weather.com.cn/data/101180601.html"]];    //将请求的url数据放到NSData对象中    NSData *response = [NSURLConnection sendSynchronousRequest:request returningResponse:nil error:nil];    //IOS5自带解析类NSJSONSerialization从response中解析出数据放到字典中    NSDictionary *weatherDic = [NSJSONSerialization JSONObjectWithData:response options:NSJSONReadingMutableLeaves error:&error];    NSDictionary *weatherInfo = [weatherDic objectForKey:@"weatherinfo"];    txtView.text = [NSString stringWithFormat:@"今天是 %@  %@  %@  的天气状况是:%@  %@ ",[weatherInfo objectForKey:@"date_y"],[weatherInfo objectForKey:@"week"],[weatherInfo objectForKey:@"city"], [weatherInfo objectForKey:@"weather1"], [weatherInfo objectForKey:@"temp1"]];    NSLog(@"weatherInfo字典里面的内容为--》%@", weatherDic );}
(4)使用JSONKit的解析方法:(需导入包:#import "JSONKit/JSONKit.h")

[cpp]
  1. - (IBAction)btnPressJsonKit:(id)sender {
  2. //如果json是“单层”的,即value都是字符串、数字,可以使用objectFromJSONString
  3. NSString *json1 = @"{\"a\":123, \"b\":\"abc\"}";
  4. NSLog(@"json1:%@",json1);
  5. NSDictionary *data1 = [json1 objectFromJSONString];
  6. NSLog(@"json1.a:%@",[data1 objectForKey:@"a"]);
  7. NSLog(@"json1.b:%@",[data1 objectForKey:@"b"]);
  8. [json1 release];
  9. //如果json有嵌套,即value里有array、object,如果再使用objectFromJSONString,程序可能会报错(测试结果表明:使用由网络或得到的php/json_encode生成的json时会报错,但使用NSString定义的json字符串时,解析成功),最好使用objectFromJSONStringWithParseOptions:
  10. NSString *json2 = @"{\"a\":123, \"b\":\"abc\", \"c\":[456, \"hello\"], \"d\":{\"name\":\"张三\", \"age\":\"32\"}}";
  11. NSLog(@"json2:%@", json2);
  12. NSDictionary *data2 = [json2 objectFromJSONStringWithParseOptions:JKParseOptionLooseUnicode];
  13. NSLog(@"json2.c:%@", [data2 objectForKey:@"c"]);
  14. NSLog(@"json2.d:%@", [data2 objectForKey:@"d"]);
  15. [json2 release];
  16. }
- (IBAction)btnPressJsonKit:(id)sender {        //如果json是“单层”的,即value都是字符串、数字,可以使用objectFromJSONString    NSString *json1 = @"{\"a\":123, \"b\":\"abc\"}";    NSLog(@"json1:%@",json1);    NSDictionary *data1 = [json1 objectFromJSONString];    NSLog(@"json1.a:%@",[data1 objectForKey:@"a"]);    NSLog(@"json1.b:%@",[data1 objectForKey:@"b"]);    [json1 release];        //如果json有嵌套,即value里有array、object,如果再使用objectFromJSONString,程序可能会报错(测试结果表明:使用由网络或得到的php/json_encode生成的json时会报错,但使用NSString定义的json字符串时,解析成功),最好使用objectFromJSONStringWithParseOptions:    NSString *json2 = @"{\"a\":123, \"b\":\"abc\", \"c\":[456, \"hello\"], \"d\":{\"name\":\"张三\", \"age\":\"32\"}}";    NSLog(@"json2:%@", json2);    NSDictionary *data2 = [json2 objectFromJSONStringWithParseOptions:JKParseOptionLooseUnicode];    NSLog(@"json2.c:%@", [data2 objectForKey:@"c"]);    NSLog(@"json2.d:%@", [data2 objectForKey:@"d"]);    [json2 release];}
另外,由于iOS5新增了JSON解析的API,我们将其和其他五个开源的JSON解析库进行了解析速度的测试,下面是测试的结果。

我们选择的测试对象包含下面的这几个框架,其中NSJSONSerialization是iOS5系统新增的JSON解析的API,需要iOS5的环境,如果您在更低的版本进行测试,应该屏蔽相应的代码调用。

- [SBJSON (json-framework)](http://code.google.com/p/json-framework/)

- [TouchJSON (from touchcode)](http://code.google.com/p/touchcode/)

- [YAJL (objective-C bindings)](http://github.com/gabriel/yajl-objc)

- [JSONKit](https://github.com/johnezang/JSONKit)

- [NextiveJson](https://github.com/nextive/NextiveJson)

-[NSJSONSerialization](http://developer.apple.com/library/ios/#documentation/Foundation/Reference/NSJSONSerialization_Class/Reference/Reference.html#//apple_ref/doc/uid/TP40010946)

我们选择了四个包含json格式的数据的文件进行测试。每一个文件进行100的解析动作,对解析的时间进行比较。

.....

测试的结果显示,系统的API的解析速度最快,我们在工程项目中选择使用,也是应用较为广泛的SBJSON的解析速度为倒数第二差,令我大跌眼镜。

与系统API较为接近的应该是JSONKit。

这里没有对API的开放接口和使用方式进行比较,若单纯基于以上解析速度的测试:

1:iOS5应该选择系统的API进行

2:不能使用系统API的应该选择JSONKit

转载地址:http://kezpi.baihongyu.com/

你可能感兴趣的文章
20个常用的JavaScript字符串方法
查看>>
【HTML 教程】图像标签
查看>>
处理 JS中 undefined 的 7 个技巧
查看>>
纯CSS3实现的几款条纹大背景
查看>>
CSS怎么实现内容不相同的左右两个div等高?
查看>>
CSS flex布局入门
查看>>
使用 ssh 连接 Github 代码库超详细教程
查看>>
21个React开发神器
查看>>
8种常见数据结构及其Javascript实现
查看>>
13 个 NPM 快速开发技巧
查看>>
8道有意思的JavaScript面试题,附解答
查看>>
月入5万的大佬们,假期一般都在干什么?看他们的公众号就知道了
查看>>
5分钟,掌握9个简洁的JavaScript技巧
查看>>
【视频教程】帝国CMS模板开发制作系列教程03
查看>>
ES6、ES7、ES8、ES9、ES10新特性一览
查看>>
JS创建对象的4种方式
查看>>
利用HTML5标签进行DDoS攻击的新方法揭秘
查看>>
CSS如何实现N宫格布局?
查看>>
如何实现Web页面录屏?
查看>>
【视频教程】帝国CMS模板开发制作网站系列教程04
查看>>