簡介
- JSON是一種輕量級(jí)的數(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必須用雙引號(hào)
- 要想從JSON中挖掘出具體數(shù)據(jù),得對JSON進(jìn)行解析
-
JSON
轉(zhuǎn)換為OC數(shù)據(jù)類型
-
JSON解析方案
在iOS中,JSON的常見解析方案有4種
第三方框架:JSONKit、SBJson、TouchJSON(性能從左到右,越差)
蘋果原生(自帶):
NSJSONSerialization(性能最好)
-
NSJSONSerialization的常見方法
- JSON數(shù)據(jù)(NSData) -> OC對象(Foundation Object)
- {} -> NSDictionary @{}
- [] -> NSArray @[]
- "jack" -> NSString @"jack"
- 10 -> NSNumber @10
- 10.5 -> NSNumber @10.5
- true -> NSNumber @1
- false -> NSNumber @0
- null -> NSNull
/* 第一個(gè)參數(shù): 需要解析的JSON數(shù)據(jù) 第二個(gè)參數(shù): 解析JSON的可選配置參數(shù) NSJSONReadingMutableContainers解析出來的字典和數(shù)組是可變的 NSJSONReadingMutableLeaves解析出來的對象中的字符串是可變的 iOS7以后有問題 NSJSONReadingAllowFragments被解析的JSON數(shù)據(jù)如果既不是字典也不是數(shù)組, 那么就必須使用這個(gè) */ + (id)JSONObjectWithData:(NSData *)data options:(NSJSONReadingOptions)opt error:(NSError **)error;
- JSON數(shù)據(jù)(NSData) -> OC對象(Foundation Object)
-
OC對象 -> JSON數(shù)據(jù)
/* 第一個(gè)參數(shù): 需要轉(zhuǎn)換為JSON數(shù)據(jù)的OC對象 第二個(gè)參數(shù): 毫無意義 NSJSONWritingPrettyPrinted:對轉(zhuǎn)換之后的JSON進(jìn)行排版 */ + (NSData *)dataWithJSONObject:(id)obj options:(NSJSONWritingOptions)opt error:(NSError **)error;
XML的解析方式有2種
DOM
:一次性將整個(gè)XML文檔加載進(jìn)內(nèi)存,比較適合解析小文件
SAX
:從根元素開始,按順序一個(gè)元素一個(gè)元素往下解析,比較適合解析大文件
SAX
(從根元素開始,按順序一個(gè)元素一個(gè)元素往下解析,比較適合解析大文件)
簡介
- 全稱是Extensible Markup Language,譯作“
可擴(kuò)展標(biāo)記語言
” - 跟JSON一樣,也是常用的一種用于交互的數(shù)據(jù)格式
- 一般也叫XML文檔(XML Document)
- 一個(gè)常見的XML文檔一般由以下部分組成
- 文檔聲明
- 元素(Element)
- 屬性(Attribute)
<videos> // 根元素
<video name="小黃人 第01部" length="30" />
<video name="小黃人 第02部" length="19" />
<video name="小黃人 第03部" length="33" />
</videos>
XML語法 – 文檔聲明
- 在XML文檔的最前面,必須編寫一個(gè)文檔聲明,用來聲明XML文檔的類型
- 最簡單的聲明
<?xml version="1.0" ?>
- 用encoding屬性說明文檔的字符編碼
<?xml version="1.0" encoding="UTF-8" ?>
XML語法 – 元素(Element)
-
一個(gè)元素包括了開始標(biāo)簽和結(jié)束標(biāo)簽
- 擁有內(nèi)容的元素:
<video>小黃人</video>
- 沒有內(nèi)容的元素:
<video></video>
- 沒有內(nèi)容的元素簡寫:
<video/>
- 擁有內(nèi)容的元素:
一個(gè)元素可以嵌套若干個(gè)子元素(不能出現(xiàn)交叉嵌套)
<videos>
<video>
<name>小黃人 第01部</name>
<length>30</length>
</video>
</videos>
- 規(guī)范的XML文檔最多只有1個(gè)根元素,其他元素都是根元素的子孫元素
- 注意:
XML中的所有空格和換行,都會(huì)當(dāng)做具體內(nèi)容處理
-
下面兩個(gè)元素的內(nèi)容是不一樣的
- 第1個(gè)
objc<video>小黃人</video>
- 第2個(gè)
<video> 小黃人 </video>
XML語法 – 屬性(Attribute)
- 一個(gè)元素可以擁有多個(gè)屬性
<video name="小黃人 第01部" length="30" /video>
- video元素?fù)碛衝ame和length兩個(gè)屬性
- 屬性值必須用 雙引號(hào)"" 或者 單引號(hào)'' 括住
- 實(shí)際上,屬性表示的信息也可以用子元素來表示,比如
<video>
<name>小黃人 第01部</name>
<length>30</length>
</video>
XML解析
-
蘋果原生
-
NSXMLParser
:SAX方式解析,使用簡單
-
-
第三方框架
-
libxml2
:純C語言,默認(rèn)包含在iOS SDK中,同時(shí)支持DOM和SAX方式解析 -
GDataXML
:DOM方式解析,由Google開發(fā),基于libxml2
-
大文件:
NSXMLParser
、libxml2小文件:
GDataXML
、NSXMLParser
、libxml2
NSXMLParser
- 使用步驟
// 傳入XML數(shù)據(jù),創(chuàng)建解析器
NSXMLParser *parser = [[NSXMLParser alloc] initWithData:data];
// 設(shè)置代理,監(jiān)聽解析過程
parser.delegate = self;
// 開始解析
[parser parse];
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
DOM
(一次性將整個(gè)XML文檔加載進(jìn)內(nèi)存,比較適合解析小文件)
GDataXML配置
-
GDataXML基于libxml2庫,得做以下配置
-
導(dǎo)入libxml2庫
設(shè)置libxml2的頭文件搜索路徑(為了能找到libxml2庫的所有頭文件)
在Head Search Path中加入/usr/include/libxml2
設(shè)置鏈接參數(shù)(自動(dòng)鏈接libxml2庫)
在Other Linker Flags中加入-lxml2
-
-
由于GDataXML是非ARC的,因此得設(shè)置編譯參數(shù)
GDataXML使用
- GDataXML中常用的類
GDataXMLDocument:代表整個(gè)XML文檔
-
GDataXMLElement
- 代表文檔中的每個(gè)元素
- 使用attributeForName:方法可以獲得屬性值
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
// 創(chuàng)建url
NSURL *url = [NSURL URLWithString:@"http://120.25.226.186:32812/video?type=XML"];
// 創(chuàng)建請求
NSURLRequest *request = [NSURLRequest requestWithURL:url];
// 發(fā)送請求
[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
// 解析數(shù)據(jù)
GDataXMLDocument *doc = [[GDataXMLDocument alloc] initWithData:data options:kNilOptions error:nil];
// 獲取根元素
GDataXMLElement *rootElement = doc.rootElement;
// 從根元素中獲取每一個(gè)子元素
NSArray *element = [rootElement elementsForName:@"video"];
// 遍歷子元素,轉(zhuǎn)換成模型
for (GDataXMLElement *ele in element) {
// 創(chuàng)建模型
SFWVideo *video = [[SFWVideo alloc] init];
// 給模型的屬性賦值
video.name = [ele attributeForName:@"name"].stringValue;
video.url = [ele attributeForName:@"url"].stringValue;
video.image = [ele attributeForName:@"image"].stringValue;
video.length = @([ele attributeForName:@"length"].stringValue.integerValue);
[self.videos addObject:video];
}
// 刷新數(shù)據(jù)
[self.tableView reloadData];
}];
}
JSON和XML比較
- 同一份數(shù)據(jù),既可以用JSON來表示,也可以用XML來表示
- 相比之下,JSON的體積小于XML,所以服務(wù)器返回給移動(dòng)端的數(shù)據(jù)格式以JSON居多