JSON
什么是JSON
JSON是一種輕量級的數(shù)據(jù)格式,一般用于數(shù)據(jù)交互.
服務(wù)器返回給客戶端的數(shù)據(jù),一般都是JSON格式或者XML格式(文件下載除外)
JSON的格式很像OC中的字典和數(shù)組
{"name" : "jack", "age" : 10}
{"names" : ["jack", "rose", "jim"]}
標(biāo)準(zhǔn)JSON格式的注意點(diǎn):key必須用雙引號要想從JSON中挖掘出具體數(shù)據(jù),得對JSON進(jìn)行解析
JSON 轉(zhuǎn)換為 OC數(shù)據(jù)類型
{} -> 字典
[] -> 數(shù)組
"" -> 字符串
10/10.1 -> NSNumber
true/false -> NSNumber
null -> NSNullJSON解析方案
在iOS中,JSON的常見解析方案有4種
第三方框架:JSONKit、SBJson、TouchJSON(性能從左到右,越差)
蘋果原生(自帶):NSJSONSerialization(性能最好)JSON數(shù)據(jù) -> OC對象
// 1.創(chuàng)建URL
NSURL *url = [NSURL URLWithString:@"xxx"];
// 2.根據(jù)URL創(chuàng)建Request
NSURLRequest *request = [NSURLRequest requestWithURL:url];
// 3.利用NSURLConnection發(fā)送請求
[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
/*
第一個(gè)參數(shù): 需要解析的JSON數(shù)據(jù)
第二個(gè)參數(shù): 解析JSON的可選配置參數(shù)
NSJSONReadingMutableContainers 解析出來的字典和數(shù)組是可變的
NSJSONReadingMutableLeaves 解析出來的對象中的字符串是可變的 iOS7以后有問題
NSJSONReadingAllowFragments 被解析的JSON數(shù)據(jù)如果既不是字典也不是數(shù)組, 那么就必須使用這個(gè)
*/
NSMutableDictionary *dict = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];
NSLog(@"%@", dict);
}];
**************** 既不是數(shù)組也不是字典 ***************
NSString *str = @"null";
//返回的既不是字典也不是數(shù)組
id obj = [NSJSONSerialization JSONObjectWithData:[str dataUsingEncoding:NSUTF8StringEncoding] options:NSJSONReadingAllowFragments error:nil];
NSLog(@"%@", [obj class]);
- OC對象 -> JSON數(shù)據(jù)
NSDictionary *dict = @{
@"name":@"YYY",
@"age":@20,
@"height":@1.75
};
/*
第一個(gè)參數(shù): 需要轉(zhuǎn)換為JSON數(shù)據(jù)的OC對象
第二個(gè)參數(shù): 毫無意義
NSJSONWritingPrettyPrinted : 對轉(zhuǎn)換之后的JSON進(jìn)行排版
*/
NSData *data = [NSJSONSerialization dataWithJSONObject:dict options:NSJSONWritingPrettyPrinted error:nil];
NSLog(@"%@", [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]);
XML
什么是XML
全稱是Extensible Markup Language,譯作“可擴(kuò)展標(biāo)記語言”
跟JSON一樣,也是常用的一種用于交互的數(shù)據(jù)格式,一般也叫XML文檔(XML Document)XML舉例
<videos>
<video name="小黃人 第01部" length="30" />
<video name="小黃人 第02部" length="19" />
<video name="小黃人 第03部" length="33" />
</videos>
- XML語法
一個(gè)常見的XML文檔一般由以下部分組成
1.文檔聲明
在XML文檔的最前面,必須編寫一個(gè)文檔聲明,用來聲明XML文檔的類型
最簡單的聲明
<?xml version="1.0" ?>
用encoding屬性說明文檔的字符編碼
<?xml version="1.0" encoding="UTF-8" ?>
2.元素(Element)
一個(gè)元素包括了開始標(biāo)簽和結(jié)束標(biāo)簽
擁有內(nèi)容的元素:<video>小黃人</video>
沒有內(nèi)容的元素:<video></video>
沒有內(nèi)容的元素簡寫:<video/>
一個(gè)元素可以嵌套若干個(gè)子元素(不能出現(xiàn)交叉嵌套)
<videos>
<video>
<name>小黃人 第01部</name>
<length>30</length>
</video>
</videos>
規(guī)范的XML文檔最多只有1個(gè)根元素,其他元素都是根元素的子孫元素
XML中的所有空格和換行,都會當(dāng)做具體內(nèi)容處理
下面兩個(gè)元素的內(nèi)容是不一樣的
第1個(gè)
<video>小黃人</video>
第2個(gè)
<video>
小黃人
</video>
3.屬性(Attribute)
一個(gè)元素可以擁有多個(gè)屬性
<video name="小黃人 第01部" length="30" />
video元素?fù)碛衝ame和length兩個(gè)屬性
屬性值必須用 雙引號"" 或者 單引號'' 括住
實(shí)際上,屬性表示的信息也可以用子元素來表示,比如
<video>
<name>小黃人 第01部</name>
<length>30</length>
</video>
XML解析
要想從XML中提取有用的信息,必須得學(xué)會解析XML
提取name元素里面的內(nèi)容
<name>小黃人 第01部</name>
提取video元素中name和length屬性的值
<video name="小黃人 第01部" length="30" />
XML的解析方式有2種
DOM:一次性將整個(gè)XML文檔加載進(jìn)內(nèi)存,比較適合解析小文件
SAX:從根元素開始,按順序一個(gè)元素一個(gè)元素往下解析,比較適合解析大文件
在iOS中,解析XML的手段有很多
蘋果原生
NSXMLParser:SAX方式解析,使用簡單
第三方框架
libxml2:純C語言,默認(rèn)包含在iOS SDK中,同時(shí)支持DOM和SAX方式解析
GDataXML:DOM方式解析,由Google開發(fā),基于libxml2
XML解析方式的選擇建議
大文件:NSXMLParser、libxml2
小文件:GDataXML、NSXMLParser、libxml2NSXMLParser解析步驟
// 傳入XML數(shù)據(jù),創(chuàng)建解析器
NSXMLParser *parser = [[NSXMLParser alloc] initWithData:data];
// 設(shè)置代理,監(jiān)聽解析過程
parser.delegate = self;
// 開始解析
[parser parse];
NSXMLParser采取的是SAX方式解析,特點(diǎn)是事件驅(qū)動,下面情況都會通知代理
當(dāng)掃描到文檔(Document)的開始與結(jié)束
當(dāng)掃描到元素(Element)的開始與結(jié)束
NSXMLParserDelegate
當(dāng)掃描到文檔的開始時(shí)調(diào)用(開始解析)
//- (void)parserDidStartDocument:(NSXMLParser *)parser//
當(dāng)掃描到文檔的結(jié)束時(shí)調(diào)用(解析完畢)
//- (void)parserDidEndDocument:(NSXMLParser *)parser
當(dāng)掃描到元素的開始時(shí)調(diào)用(attributeDict存放著元素的屬性)
//- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict
當(dāng)掃描到元素的結(jié)束時(shí)調(diào)用
//- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName
GDataXML配置
- GDataXML解析步驟
GDataXML基于libxml2庫,得做以下配置
1.導(dǎo)入libxml2庫
2.設(shè)置libxml2的頭文件搜索路徑(為了能找到libxml2庫的所有頭文件)
在Head Search Path中加入/usr/include/libxml2
3.設(shè)置鏈接參數(shù)(自動鏈接libxml2庫)
在Other Linker Flags中加入-lxml2
// 1.創(chuàng)建URL
NSURL *url = [NSURL URLWithString:@"http://xxx?type=XML"];
// 2.根據(jù)URL創(chuàng)建Request
NSURLRequest *request = [NSURLRequest requestWithURL:url];
// 3.利用NSURLConnection發(fā)送請求
[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
// 1.加載所有的xml到內(nèi)存中
GDataXMLDocument *doc = [[GDataXMLDocument alloc] initWithData:data options:kNilOptions error:nil];
// 2.獲取根元素
GDataXMLElement *rootElement = doc.rootElement;
// 3.從根元素中獲取所有子元素
NSArray *elements = [rootElement elementsForName:@"video"];
// 4.將子元素中的屬性轉(zhuǎn)換為模型
for (GDataXMLElement *ele in elements) {
YYYVideo *video = [[XMGVideo alloc] init];
video.image = [ele attributeForName:@"image"].stringValue;
video.url = [ele attributeForName:@"url"].stringValue;
video.name = [ele attributeForName:@"name"].stringValue;
video.length = @([ele attributeForName:@"length"].stringValue.integerValue);
[self.videos addObject:video];