iOS中發(fā)送http請求的方案
- 蘋果原生
- NSURLConnection 03年推出的古老技術(shù)
- NSURLSession 13年推出iOS7之后疚顷,以取代NSURLConnection
- CFNetwork 底層技術(shù)棚放、C語言的
- 第三方框架
- ASIHttpRequest
- AFNetworking
- MKNetworkKit
NSURLConnection使用
- NSURLConnection同步get請求
- 步驟
- 設(shè)置請求路徑
- 創(chuàng)建請求對象(默認(rèn)是GET請求,且已經(jīng)默認(rèn)包含了請求頭)
- 使用NSURLConnection sendsync方法發(fā)送網(wǎng)絡(luò)請求
- 接收到服務(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-代理)
- 步驟
- 確定請求路徑
- 創(chuàng)建請求對象
- 創(chuàng)建NSURLConnection對象并設(shè)置代理
- 遵守NSURLConnectionDataDelegate協(xié)議,并實現(xiàn)相應(yīng)的代理方法
- 在代理方法中監(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請求步驟
- 確定URL路徑
- 創(chuàng)建請求對象(可變對象)
- 修改請求對象的方法為POST赐俗,設(shè)置請求體(Data)
- 發(fā)送一個異步請求
- 補(bǔ)充:設(shè)置請求超時,處理錯誤信息弊知,設(shè)置請求頭(如獲取客戶端的版本等等,請求頭是可設(shè)置可不設(shè)置的)
- 代碼
- 發(fā)送POST請求步驟
//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];