iOS網(wǎng)絡編程--基礎使用

iOS中發(fā)送http請求的方案

  • 蘋果原生
    • NSURLConnection 03年推出的古老技術
    • NSURLSession 13年推出iOS7之后,以取代NSURLConnection
    • CFNetwork 底層技術、C語言的
  • 第三方框架
    • ASIHttpRequest
    • AFNetworking
    • MKNetworkKit

NSURLConnection使用

  • NSURLConnection同步get請求
    • 步驟
      1. 設置請求路徑
      2. 創(chuàng)建請求對象(默認是GET請求,且已經(jīng)默認包含了請求頭)
      3. 使用NSURLConnection sendsync方法發(fā)送網(wǎng)絡請求
      4. 接收到服務器的響應后,解析響應體
    • 代碼
//1.確定請求路徑 
NSURL *url = [NSURL URLWithString:@"URLString(發(fā)送請求的地址路徑)"]; 
//2.創(chuàng)建一個請求對象
NSURLRequest *request = [NSURLRequest requestWithURL:url];
//3.把請求發(fā)送給服務器
//sendSynchronousRequest 阻塞式的方法,會卡住線程
NSHTTPURLResponse *response = nil;NSError *error = nil;
/* 
第一個參數(shù):請求對象 
第二個參數(shù):響應頭信息,當該方法執(zhí)行完畢之后,該參數(shù)被賦值 
第三個參數(shù):錯誤信息,如果請求失敗,則error有值 
*/ 
//該方法是阻塞式的,會卡住線程
NSData *data = [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error];
//4.解析服務器返回的數(shù)據(jù)
NSString *str = [[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding];
  • NSURLConnection異步請求(get-sendAsync)
    該方法不會卡住線程,網(wǎng)絡請求任務是異步執(zhí)行的
//1.確定請求路徑 
NSURL *url = [NSURL URLWithString:@"URLString(發(fā)送請求的地址路徑)"]; 
//2.創(chuàng)建一個請求對象 
NSURLRequest *request = [NSURLRequest requestWithURL:url]; 
//3.把請求發(fā)送給服務器,發(fā)送一個異步請求 
/* 
第一個參數(shù):請求對象 
第二個參數(shù):回調(diào)方法在哪個線程中執(zhí)行,如果是主隊列則block在主線程中執(zhí)行,非主隊列則在子線程中執(zhí)行 
第三個參數(shù):completionHandlerBlock塊:接受到響應的時候執(zhí)行該block中的代碼 response:響應頭信息 data:響應體 connectionError:錯誤信息,如果請求失敗,那么該參數(shù)有值 
*/ 
[NSURLConnection sendAsynchronousRequest:request queue:[[NSOperationQueue alloc]init] completionHandler:^(NSURLResponse * __nullable response, NSData * __nullable data, NSError * __nullable connectionError) { 
      //4.解析服務器返回的數(shù)據(jù) 
      NSString *str = [[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding]; 
      //轉(zhuǎn)換并打印響應頭信息 
      NSHTTPURLResponse *r = (NSHTTPURLResponse *)response; 
      NSLog(@"--%zd---%@--",r.statusCode,r.allHeaderFields); 
}];
  • NSURLConnection異步請求(get-代理)
    • 步驟
      1. 確定請求路徑
      2. 創(chuàng)建請求對象
      3. 創(chuàng)建NSURLConnection對象并設置代理
      4. 遵守NSURLConnectionDataDelegate協(xié)議,并實現(xiàn)相應的代理方法
      5. 在代理方法中監(jiān)聽網(wǎng)絡請求的響應
    • 設置代理的幾種方式
/* 
設置代理的第一種方式:自動發(fā)送網(wǎng)絡請求 
[[NSURLConnection alloc]initWithRequest:request delegate:self]; 
*/
/* 
設置代理的第二種方式: 
第一個參數(shù):請求對象 
第二個參數(shù):誰成為NSURLConnetion對象的代理
第三個參數(shù):是否馬上發(fā)送網(wǎng)絡請求,如果該值為YES則立刻發(fā)送,如果為NO則不會發(fā)送網(wǎng)路請求 
NSURLConnection *conn = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:NO]; 
//調(diào)用該方法控制網(wǎng)絡請求的發(fā)送 
//注意該方法內(nèi)部會自動的把connect添加到當前線程的RunLoop中在默認模式下執(zhí)行
[conn start]; 
*/
//設置代理的第三種方式:使用類方法設置代理,會自動發(fā)送網(wǎng)絡請求
NSURLConnection *conn = [NSURLConnection connectionWithRequest:request delegate:self];
//取消網(wǎng)絡請求
//[conn cancel];
  • 相關代理方法
/* 
1.當接收到服務器響應的時候調(diào)用 
第一個參數(shù)connection:監(jiān)聽的是哪個NSURLConnection對象 
第二個參數(shù)response:接收到的服務器返回的響應頭信息 
*/ 
-(void)connection:(nonnull NSURLConnection *)connection didReceiveResponse:(nonnull NSURLResponse *)response 
/* 
2.當接收到數(shù)據(jù)的時候調(diào)用,該方法會被調(diào)用多次 
第一個參數(shù)connection:監(jiān)聽的是哪個NSURLConnection對象 
第二個參數(shù)data:本次接收到的服務端返回的二進制數(shù)據(jù)(可能是片段) 
*/ 
-(void)connection:(nonnull NSURLConnection *)connection didReceiveData:(nonnull NSData *)data 
/* 
3.當服務端返回的數(shù)據(jù)接收完畢之后會調(diào)用 通常在該方法中解析服務器返回的數(shù)據(jù) */
 -(void)connectionDidFinishLoading:(nonnull NSURLConnection *)connection 
/*
4.當請求錯誤的時候調(diào)用(比如請求超時) 
第一個參數(shù)connection:NSURLConnection對象 
第二個參數(shù):網(wǎng)絡請求的錯誤信息,如果請求失敗,則error有值 
*/ 
-(void)connection:(nonnull NSURLConnection *)connection didFailWithError:(nonnull NSError *)error
  • 如何控制代理方法在哪個線程調(diào)用*****
//說明:默認情況下,代理方法會在主線程中進行調(diào)用(為了方便開發(fā)者拿到數(shù)據(jù)后處理一些刷新UI的操作不需要考慮到線程間通信) 
//設置代理方法的執(zhí)行隊列 
[connect setDelegateQueue:[[NSOperationQueue alloc]init]];
  • 開子線程發(fā)送網(wǎng)絡請求的注意點,適用于自動發(fā)送網(wǎng)絡請求模式
//使用GCD開啟一個子線程來發(fā)送網(wǎng)絡請求 
dispatch_async(dispatch_get_global_queue(0, 0), ^{ 
//使用非自動發(fā)送網(wǎng)絡請求模式,發(fā)送請求OK 
/* 
//創(chuàng)建NSURLConnection對象,設置代理,暫不發(fā)送 
NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:NO]; 
//設置代理方法的執(zhí)行隊列 [connect setDelegateQueue:[[NSOperationQueue alloc]init]]; 
//調(diào)用start發(fā)送網(wǎng)絡請求 
[connect start]; 
*/ 
//使用自動發(fā)送網(wǎng)絡請求模式,發(fā)送請求失敗(需要改造代碼) 
//WHY? 
/*
01 網(wǎng)絡請求發(fā)送和數(shù)據(jù)接收是否成功,和一些因素相關,比如客戶端的網(wǎng)速、服務器端的查詢速度等等。 
02 而在子線程中創(chuàng)建的NSURLConnection對象是一個臨時變量,當請求發(fā)送完成之后就被釋放了,所以這個時候它的代理方法不會調(diào)用用。 
03 為什么使用非自動發(fā)送網(wǎng)絡請求模式是OK的。 因為在該模式中,調(diào)用了start來開始發(fā)送網(wǎng)絡請求,該方法內(nèi)部會自動將當前的connect作為一個Source添加到當前線程所在的Runloop中,如果當前線程是子線程(即當前線程的runloop并未創(chuàng)建),那么該方法內(nèi)部會默認先創(chuàng)建當前線程的Runloop,設置在runloop的默認模式下運行。 此時runloop會對這個Connect對象進行強引用,保證了代理方法被調(diào)用的前提 
*/ 
NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self]; 
[connect setDelegateQueue:[[NSOperationQueue alloc]init]]; 
//創(chuàng)建當前線程的runloop,并開啟runloop 
[[NSRunLoop currentRunLoop] run]; 
});
  • 其他知識
01 關于消息彈窗第三方框架的使用 
SVProgressHUD 
02 字符串截取相關方法 
-(NSRange)rangeOfString:(NSString *)searchString; 
-(NSString *)substringWithRange:(NSRange)range;
  • NSURLConnection發(fā)送POST請求
    • 發(fā)送POST請求步驟
      1. 確定URL路徑
      2. 創(chuàng)建請求對象(可變對象)
      3. 修改請求對象的方法為POST,設置請求體(Data)
      4. 發(fā)送一個異步請求
      5. 補充:設置請求超時,處理錯誤信息,設置請求頭(如獲取客戶端的版本等等,請求頭是可設置可不設置的)
    • 代碼
//1.確定請求路徑 
NSURL *url = [NSURL URLWithString:@"請求路徑字符串"];
//2.創(chuàng)建請求對象
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
//2.1更改請求方法
request.HTTPMethod = @"POST";
//2.2設置請求體
request.HTTPBody = [@"username=111&pwd=222" dataUsingEncoding:NSUTF8StringEncoding];
//2.3請求超時
request.timeoutInterval = 5;
//2.4設置請求頭
[request setValue:@"ios 9.0" forHTTPHeaderField:@"User-Agent"];
//3.發(fā)送請求
[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse * __nullable response, NSData * __nullable data, NSError * __nullable connectionError) 
{ 
          //4.解析服務器返回的數(shù)據(jù) 
          if (connectionError) { 
              NSLog(@"--請求失敗-"); 
          }else { 
              NSLog(@"%@",[[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding]);
          }
}];

NSURLSession使用

  • 使用步驟
    • 使用NSURLSession創(chuàng)建task,然后執(zhí)行task
  • 關于task
    • NSURLSessionTask是一個抽象類,本身不能使用,只能使用它的子類
    • NSURLSessionDataTask\NSURLSessionUploadTask\NSURLSessionDownloadTask
  • 發(fā)送get請求
//1.創(chuàng)建NSURLSession對象(可以獲取單例對象) 
NSURLSession *session = [NSURLSession sharedSession]; 
//2.根據(jù)NSURLSession對象創(chuàng)建一個Task 
NSURL *url = [NSURL URLWithString:@"URL字符串"]; 
NSURLRequest *request = [NSURLRequest requestWithURL:url]; 
//方法參數(shù)說明 
/* 
注意:該block是在子線程中調(diào)用的,如果拿到數(shù)據(jù)之后要做一些UI刷新操作,那么需要回到主線程刷新 
第一個參數(shù):需要發(fā)送的請求對象 
block:當請求結束拿到服務器響應的數(shù)據(jù)時調(diào)用block 
block-NSData:該請求的響應體 
block-NSURLResponse:存放本次請求的響應信息,響應頭,真實類型為NSHTTPURLResponse 
block-NSErroe:請求錯誤信息 
*/ 
NSURLSessionDataTask * dataTask = [session dataTaskWithRequest:request completionHandler:^(NSData * __nullable data, NSURLResponse * __nullable response, NSError * __nullable error) { 
//拿到響應頭信息 
NSHTTPURLResponse *res = (NSHTTPURLResponse *)response; 
//4.解析拿到的響應數(shù)據(jù) 
NSLog(@"%@\n%@",[[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding],res.allHeaderFields); 
}]; 
//3.執(zhí)行Task 
//注意:剛創(chuàng)建出來的task默認是掛起狀態(tài)的,需要調(diào)用該方法來啟動任務(執(zhí)行任務) 
[dataTask resume];
  • 發(fā)送get請求的第二種方式
//注意:該方法內(nèi)部默認會把URL對象包裝成一個NSURLRequest對象(默認是GET請求) 
//方法參數(shù)說明 
/* 
//第一個參數(shù):發(fā)送請求的URL地址 
//block:當請求結束拿到服務器響應的數(shù)據(jù)時調(diào)用block 
//block-NSData:該請求的響應體 
//block-NSURLResponse:存放本次請求的響應信息,響應頭,真實類型為NSHTTPURLResponse 
//block-NSErroe:請求錯誤信息 
*/ 
-(nullable NSURLSessionDataTask *)dataTaskWithURL:(NSURL *)url completionHandler:(void (^)(NSData * __nullable data, NSURLResponse * __nullable response, NSError * __nullable error))completionHandler;
  • 發(fā)送post請求
//1.創(chuàng)建NSURLSession對象(可以獲取單例對象) 
NSURLSession *session = [NSURLSession sharedSession]; 
//2.根據(jù)NSURLSession對象創(chuàng)建一個Task 
NSURL *url = [NSURL URLWithString:@"URL字符串"]; 
//創(chuàng)建一個請求對象,并這是請求方法為POST,把參數(shù)放在請求體中傳遞 NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url]; 
request.HTTPMethod = @"POST"; 
request.HTTPBody = [@"username=111&pwd=222&type=JSON" dataUsingEncoding:NSUTF8StringEncoding]; 
NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request completionHandler:^(NSData * __nullable data, NSURLResponse * __nullable response, NSError * __nullable error) { 
//拿到響應頭信息 
NSHTTPURLResponse *res = (NSHTTPURLResponse *)response; 
//解析拿到的響應數(shù)據(jù) 
NSLog(@"%@\n%@",[[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding],res.allHeaderFields); 
}]; 
//3.執(zhí)行Task 
//注意:剛創(chuàng)建出來的task默認是掛起狀態(tài)的,需要調(diào)用該方法來啟動任務(執(zhí)行任務) 
[dataTask resume];
最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,182評論 6 543
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,489評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,290評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,776評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,510評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,866評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,860評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 43,036評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,585評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,331評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,536評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,058評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,754評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,154評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,469評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,273評論 3 399
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,505評論 2 379

推薦閱讀更多精彩內(nèi)容