iOS網(wǎng)絡(luò)編程--基礎(chǔ)使用

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

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

NSURLConnection使用

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

NSURLSession使用

  • 使用步驟
    • 使用NSURLSession創(chuàng)建task阻逮,然后執(zhí)行task
  • 關(guān)于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:當(dāng)請求結(jié)束拿到服務(wù)器響應(yīng)的數(shù)據(jù)時調(diào)用block 
block-NSData:該請求的響應(yīng)體 
block-NSURLResponse:存放本次請求的響應(yīng)信息漫雷,響應(yīng)頭瓜富,真實類型為NSHTTPURLResponse 
block-NSErroe:請求錯誤信息 
*/ 
NSURLSessionDataTask * dataTask = [session dataTaskWithRequest:request completionHandler:^(NSData * __nullable data, NSURLResponse * __nullable response, NSError * __nullable error) { 
//拿到響應(yīng)頭信息 
NSHTTPURLResponse *res = (NSHTTPURLResponse *)response; 
//4.解析拿到的響應(yīng)數(shù)據(jù) 
NSLog(@"%@\n%@",[[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding],res.allHeaderFields); 
}]; 
//3.執(zhí)行Task 
//注意:剛創(chuàng)建出來的task默認(rèn)是掛起狀態(tài)的,需要調(diào)用該方法來啟動任務(wù)(執(zhí)行任務(wù)) 
[dataTask resume];
  • 發(fā)送get請求的第二種方式
//注意:該方法內(nèi)部默認(rèn)會把URL對象包裝成一個NSURLRequest對象(默認(rèn)是GET請求) 
//方法參數(shù)說明 
/* 
//第一個參數(shù):發(fā)送請求的URL地址 
//block:當(dāng)請求結(jié)束拿到服務(wù)器響應(yīng)的數(shù)據(jù)時調(diào)用block 
//block-NSData:該請求的響應(yīng)體 
//block-NSURLResponse:存放本次請求的響應(yīng)信息珊拼,響應(yīng)頭食呻,真實類型為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) { 
//拿到響應(yīng)頭信息 
NSHTTPURLResponse *res = (NSHTTPURLResponse *)response; 
//解析拿到的響應(yīng)數(shù)據(jù) 
NSLog(@"%@\n%@",[[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding],res.allHeaderFields); 
}]; 
//3.執(zhí)行Task 
//注意:剛創(chuàng)建出來的task默認(rèn)是掛起狀態(tài)的仅胞,需要調(diào)用該方法來啟動任務(wù)(執(zhí)行任務(wù)) 
[dataTask resume];
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市剑辫,隨后出現(xiàn)的幾起案子干旧,更是在濱河造成了極大的恐慌,老刑警劉巖妹蔽,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件椎眯,死亡現(xiàn)場離奇詭異,居然都是意外死亡胳岂,警方通過查閱死者的電腦和手機(jī)编整,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來乳丰,“玉大人掌测,你說我怎么就攤上這事〔埃” “怎么了汞斧?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長什燕。 經(jīng)常有香客問我粘勒,道長,這世上最難降的妖魔是什么屎即? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任庙睡,我火速辦了婚禮,結(jié)果婚禮上技俐,老公的妹妹穿的比我還像新娘埃撵。我一直安慰自己,他們只是感情好虽另,可當(dāng)我...
    茶點故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布暂刘。 她就那樣靜靜地躺著,像睡著了一般捂刺。 火紅的嫁衣襯著肌膚如雪谣拣。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天族展,我揣著相機(jī)與錄音森缠,去河邊找鬼。 笑死仪缸,一個胖子當(dāng)著我的面吹牛贵涵,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼宾茂,長吁一口氣:“原來是場噩夢啊……” “哼瓷马!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起跨晴,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤欧聘,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后端盆,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體怀骤,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年焕妙,在試婚紗的時候發(fā)現(xiàn)自己被綠了蒋伦。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,040評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡焚鹊,死狀恐怖痕届,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情寺旺,我是刑警寧澤爷抓,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站阻塑,受9級特大地震影響蓝撇,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜陈莽,卻給世界環(huán)境...
    茶點故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一渤昌、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧走搁,春花似錦独柑、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至曲稼,卻和暖如春索绪,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背贫悄。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工瑞驱, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人窄坦。 一個月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓唤反,卻偏偏與公主長得像凳寺,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子彤侍,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,979評論 2 355

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