http://www.baidu.com?city=beijing&weather=?
前面部分被稱為請求頭 ?后面的部分被稱為請求體
請求體由參數和參數值組成 多個參數之間用&連接
<1>按照接口劃分:
如果接口是完整(包括請求頭和請求體兩部分)
這種數據請求 我們就使用GET方式請求數據
GET方式請求數據非常簡單 但是不安全 請求數據的信息都在接口的請求體位置裸露出來 所以不安全
如果接口不完整(只有請求頭 沒有請求體)
這種數據請求的方式 我們就使用POST方式請求數據
POST方式請求數據 相對來說復雜一些 請求體需要寫代碼手動拼接
POST請求數據的方式經常用于登錄、注冊、文件的上傳
<2>按照請求數據的樣式劃分:
同步請求:
多個請求對象向服務器請求數據 服務器會將多個對象放在隊列中 一個對象開始請求數據 其他對象在隊列中等待 如果第一個對象請求數據結束以后 第二個對象才能開始請求數據 即使對象請求數據結束也要在隊列中等待 等待所有對象全部請求數據完成 服務器才將所有對象請求的數據反饋給UI 所有用戶體驗非常不好 經常出現白屏獲取卡頓 (也就是影響了UI主線程的工作)
異步請求:
多個對象請求數據 也是將多個對象放在隊列中 多個對象同時請求數據 誰先請求數據完成 就將數據反饋給UI 無需等待其他對象完成數據請求
iOS開發中用到的數據請求都是異步請求 異步請求能夠提高CPU的利用率
真正開發中和數據有關的就是下面的4步
1、數據請求
<1>系統類進行數據請求NSURLConnection
<2>第三方庫ASIHTTPRequest (過時)
<3>第三方庫AFNetworking (盛行)
<4>系統類NSURLSession (對NSURLConnection的二次封裝)
2、將請求下來的數據進行解析
JSON (系統類 NSJSONSerialization)
XML (系統類 NSXMLParser 第三方庫GData)
3、將解析好的數據顯示在UI上
4、將UI上的數據進行持久性存儲
NSFileHandle PList/userdefaults 歸檔 數據庫 CoreData
NSURLConnection
進行數據請求 只能做單任務異步請求
將一個請求任務分給多個線程 多個線程同時工作 互不干擾 一個線程只能完成當前任務的一部分 只有所有線程全部完成 整個任務才算結束
使用NSURLConnection進行數據異步請求 需要當前類遵守協議NSURLConnectionDataDelegate
AFNetworking
AFNetWorking 進行數據請求 不僅能請求json數據也能請求xml數據 同時代有兩種數據的自動解析
AFNetWorking對NSJSONSerialization和NSXMLParser進行了二次封裝
網絡請求數據(兩種形式)
形式一:GET方法
方法一:
NSURLConnection:(需要當前類遵守協議NSURLConnectionDataDelegate)
<1>將字符串網址轉化成NSURL
NSURL * url = [NSURL URLWithString:PATH];
<2>將NSURL封裝成請求對象
NSURLRequest * request = [NSURLRequest requestWithURL:url];
<3>開始異步請求
[NSURLConnection connectionWithRequest:request delegate:self];
協議方法
服務器端接收客戶端的信息反饋
-(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response;
開始請求數據
-(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
每個線程請求完數據以后就會調用該方法 將請求下來的數據進行拼接
[myData appendData:data];
請求數據結束
-(void)connectionDidFinishLoading:(NSURLConnection *)connection{
開始解析數據
NSDictionary * dic = [NSJSONSerialization JSONObjectWithData:myData options:NSJSONReadingMutableContainers error:nil];
NSArray * array = dic[@"users"];
[dataSource addObjectsFromArray:array];
//數據源發生改變 刷新表格
[table reloadData];
}
方法二 AFNetWorking
<1>創建請求操作管理者對象
AFHTTPRequestOperationManager * manager = [AFHTTPRequestOperationManager manager];
<2>設置請求的數據類型(是json還是xml)
manager.responseSerializer.acceptableContentTypes = [NSSet setWithObject:@"application/json"];
<3>開始數據請求并解析
1、請求數據的接口
2、請求體
3、請求數據成功、數據解析也完成了
[manager GET:<#(NSString *)#> parameters:<#(id)#> success:<#^(AFHTTPRequestOperation *operation, id responseObject)success#> failure:<#^(AFHTTPRequestOperation *operation, NSError *error)failure#>]
AFNetworking可以進行多任務異步請求
一個界面請求數據的接口有多個
只要請求一個接口的數據 就要有一個請求對象
此處需要創建四個請求對象 對象之間請求數據是異步的 互不干擾
將多個對象放在隊列中 進行異步請求
所有圖片的接口信息
NSArray * array = @[]所有接口信息
定義一個數組 承載四個請求對象
NSMutableArray * requestArr = [[NSMutableArray alloc]init];
for(int i = 0;i<4;i++)
{
//<1>將接口轉化成NSURL
NSURL * url = [NSURL URLWithString:array[i]];
//<2>將NSURL封裝成請求對象
NSURLRequest * request = [NSURLRequest requestWithURL:url];
//<3>將NSURLRequest對象封裝成AF中的請求操作對象
AFHTTPRequestOperation * operation = [[AFHTTPRequestOperation alloc]initWithRequest:request];
//<4>將請求下來的數據顯示在UI上
[operation setCompletionBlockWithSuccess:^(AFHTTPRequestOperation *operation, id responseObject) {
//【注意】對象請求數據 內部是同步進行數據請求的
UIImage * image = [UIImage imageWithData:responseObject];
UIImageView * imageView = (UIImageView *)[self.view viewWithTag:i + 1];
imageView.image = image;
} failure:^(AFHTTPRequestOperation *operation, NSError *error) {
NSLog(@"%@",error.description);
}];
[requestArr addObject:operation];
}
創建隊列對象
NSOperationQueue * queue = [[NSOperationQueue alloc]init];
設置隊列中同一時刻請求數據的對象個數
queue.maxConcurrentOperationCount = 4;
將所有請求對象放在隊列中
[queue addOperations:requestArr waitUntilFinished:NO];
形式二:POST方法(POST請求 經常用于登錄、注冊、文件上傳等操作)
方法一:
NSURLConnection:(需要當前類遵守協議NSURLConnectionDataDelegate實現協議中必須實現的三個方法)
<1>將字符串網址轉化成NSURL
NSURL * url = [NSURL URLWithString:PATH];
<2>封裝請求對象
[注意]GET請求 封裝的請求對象類型為NSURLRequest
POST請求 封裝的請求對象類型為NSMutableURLRequest
NSMutableURLRequest * request = [NSMutableURLRequest requestWithURL:url];
<3>設置請求方式
默認請求的方式是GET請求
[request setHTTPMethod:@"Post"]
<4>設置請求的樣式
當前的請求體樣式 參數名=參數值&參數名=參數值...
[request setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"];
<5>拼接請求體
NSString * bodyStr = [NSString stringWithFormat:@"username=%@&password=%@&email=%@",self.nameTextField.text,self.pswTextField.text,self.emailTextField.text];
<6>設置請求體的長度
[注意]設置請求體的長度 設置的是請求體字符串對應的NSData的長度
NSData * bodyData = [bodyStr dataUsingEncoding:NSUTF8StringEncoding];
[request setValue:[NSString stringWithFormat:@"%d",(int)bodyData.length] forHTTPHeaderField:@"Content-Length"];
<7>將請求體拼接到請求頭后面
[request setHTTPBody:bodyData];
<8>開始請求數據
[NSURLConnection connectionWithRequest:request delegate:self];
方法二 AFNetWorking
//<1>創建請求操作管理這對象
AFHTTPRequestOperationManager * manager = [AFHTTPRequestOperationManager manager];
<2>設置接收的數據類型
manager.responseSerializer.acceptableContentTypes = [NSSet setWithObject:@"text/html"];
<3>拼接請求體
AFNetworking進行post請求 請求體拼接使用字典封裝
字典的鍵為接口文檔中的參數名 字典中的值為參數值
NSDictionary * dic = @{@"username":self.nameTextField.text,@"password":self.pswTextField.text};
<4>開始請求數據并解析
[manager POST:LOGINPATH parameters:dic success:^(AFHTTPRequestOperation * _Nonnull operation, id _Nonnull responseObject)
{
NSString * message = responseObject[@"message"];
NSString * code = responseObject[@"code"];
} failure:^(AFHTTPRequestOperation * _Nonnull operation, NSError * _Nonnull error) {
NSLog(@"%@",error.description);
}];