iOS網絡基礎——NSURLConnection使用詳解舉例

一、整體介紹

NSURLConnection是蘋果提供的原生網絡訪問類,但是蘋果很快會將其廢棄,且由NSURLSession(iOS7以后)來替代。目前使用最廣泛的第三方網絡框架AFNetworking最新版本已棄用了NSURLConnection,那我們學習它還有什么用呢?

  • 首先,蘋果棄用它還是需要時間的,最起碼到iOS10之后;
  • 現在還有一些老項目會使用NSURLConnection,特別是2013年之前的項目,用戶量基礎還是很大的;
  • 另外,不得不承認,有些公司還在用類似ASI這些經典的網絡框架,所以還是很有必要學習NSURLConnection的。

二、使用的一般步驟

1 NSURL:請求地址,定義一個網絡資源路徑:

NSURL *url = [NSURL URLWithString:@"協議://主機地址/路徑?參數&參數"];

解釋如下:

  • 協議:不同的協議,代表著不同的資源查找方式、資源傳輸方式,比如常用的http,ftp等
  • 主機地址:存放資源的主機的IP地址(域名)
  • 路徑:資源在主機中的具體位置
  • 參數:參數可有可無,也可以多個。如果帶參數的話,用“?”號后面接參數,多個參數的話之間用&隔開

2 NSURLRequest:請求,根據前面的NSURL建立一個請求:

NSURLRequest *request = [NSURLRequest requestWithURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:15.0];

參數解釋如下:

  • url:資源路徑
  • cachePolicy:緩存策略(無論使用哪種緩存策略,都會在本地緩存數據),類型為美劇類型,取值如下:
    • NSURLRequestUseProtocolCachePolicy = 0 //默認的緩存策略,使用協議的緩存策略
    • NSURLRequestReloadIgnoringLocalCacheData = 1 //每次都從網絡加載
    • NSURLRequestReturnCacheDataElseLoad = 2 //返回緩存否則加載,很少使用
    • NSURLRequestReturnCacheDataDontLoad = 3 //只返回緩存,沒有也不加載,很少使用
  • timeoutInterval:超時時長,默認60s

另外,還可以設置其它一些信息,比如請求頭,請求體等等,如下:

注意,下面的request應為NSMutableURLRequest,即可變類型

// 告訴服務器數據為json類型
[request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"]; 
// 設置請求體(json類型)
NSData *jsonData = [NSJSONSerialization dataWithJSONObject:@{@"userid":@"123456"} options:NSJSONWritingPrettyPrinted error:nil];
request.HTTPBody = jsonData; 

3 發送請求:

NSURLConnection默認的請求類型為GET,下面分異步和同步兩種介紹

異步請求
[NSURLConnection sendAsynchronousRequest:request queue:[[NSOperationQueue alloc] init] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
    // 有的時候,服務器訪問正常,但是會沒有數據!
    // 以下的 if 是比較標準的錯誤 處理代碼!
    if (connectionError != nil || data == nil) {
        //給用戶的提示信息
        NSLog(@"網絡不給力");
        return;
    }
}];

參數說明如下:

  • completionHandler:請求響應后(或者請求超時)執行的代碼,queue為代碼添加到的隊列,即block執行的線程
    • NSURLResponse 為服務器的響應,真實類型為NSHTTPURLResponse,通常只在“下載”功能時,才會使用;下面是協議頭的參數:
      • URL:響應的URL,有的時候,訪問一個URL地址,服務器可能會出現重定向,會定位到新的地址!
      • MIMEType(Content-Type):服務器告訴客戶端,可以用什么軟件打開二進制數據!網絡之所以豐富多采,是因為有豐富的客戶端軟件!栗子:windows上提示安裝 Flash 插件
      • expectedContentLength:預期的內容長度,要下載的文件長度,下載文件時非常有用
      • suggestedFilename:"建議"的文件名,方便用戶直接保存,很多時候,用戶并不關心要保存成什么名字!
      • textEncodingName:文本的編碼名稱 @"UTF8",大多數都是 UTF8
      • statusCode:狀態碼,在做下載操作的時候,需要判斷一下
      • allHeaderFields:所有的響應頭字典時候,用戶并不關心要保存成什么名字!
    • NSData 服務器返回的數據,例如json、xml(現在用的少)
    • NSError 網絡訪問錯誤碼

注意,block的執行線程為queue,如果block涉及到UI操作,則必須回到主線程:[NSOperationQueue mainQueue]

發送同步請求
// 同步請求,代碼會阻塞在這里一直等待服務器返回,如果data為nil則請求失敗,當獲取少量數據時可以使用此方法。
// request參數同上。
NSData *data = [NSURLConnection sendSynchronousRequest:request returningResponse:nil error:nil]; 

三、舉例說明

分三個部分:

  1. 網絡請求(json、xml數據)
  2. 文件下載
  3. 文件上傳,這里例子不再給出,請參考這里:文件上傳

1 一般的網絡請求

/// 兩種請求方式
typedef enum {
    MethodGET,
    MethodPOST
}Method;

/// 請求成功后,直接將jsonData轉為jsonDict
+ (void)connectionRequestWithMethod:(Method)method   URLString:(NSString *)URLString parameters:(NSDictionary *)dict success:(void (^)(id JSON))success fail:(void (^)(NSError *error))fail {
    URLString = [URLString stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    // 1.創建請求
    NSURL *url = [NSURL URLWithString:URLString];
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    // 請求方式,默認為GET
    if (method == MethodPOST) {
        request.HTTPMethod = @"POST";
    }
    // 根據需要設置
    [request setValue:@"text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8" forHTTPHeaderField:@"Accept"];
    
    // 2.設置請求頭 Content-Type  返回格式
    [request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
    
    // 3.設置請求體 NSDictionary --> NSData
    if (dict != nil) {
        NSData *data = [NSJSONSerialization dataWithJSONObject:dict options:NSJSONWritingPrettyPrinted error:nil];
        request.HTTPBody = data;
    }
    // 4.發送請求
    [NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
        if ((data != nil) && (connectionError == nil)) {
            NSDictionary *jsonDict = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];
            if (success) {
                success(jsonDict);
            }
        } else {
            if (fail) {
                fail(connectionError);
            }
        }
        
    }];
}

2 文件下載

下面舉一個實現斷點續傳的網絡下載,要實現斷點續傳,還需要用到NSURLConnectionDataDelegate代理,NSFileHandle文件操作或者數據流的形式(NSOutputStream),這里以NSFileHandle為例,訪問的服務器采用Apache搭建的本地服務器,具體見代碼:

@interface ViewController () <NSURLConnectionDataDelegate>

/// 文件下載完畢之后,在本地保存的路徑
@property (nonatomic, copy) NSString *filePath;
/// 需要下載的文件的總大小!
@property (nonatomic, assign) long long serverFileLength;
/// 當前已經下載長度
@property (nonatomic, assign) long long localFileLength;
@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];    
}
/// 點擊屏幕事件
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    [self downloadFile];
}
/// 下載文件
- (void)downloadFile {
    // 文件存儲路徑
    self.filePath = @"/Users/username/Desktop//陶喆 - 愛很簡單.mp3";
    // 要下載的網絡文件,采用Apache搭建的本地服務器
    NSString *urlStr = @"http://localhost/陶喆 - 愛很簡單.mp3";
    urlStr = [urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
    NSURL *url = [NSURL URLWithString:urlStr];
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    // 設置請求方法為 HEAD 方法,這里只是頭,數據量少,用同步請求也可以,不過最好還是用異步請求
    request.HTTPMethod = @"HEAD";
    // 無論是否會引起循環引用,block里面都使用弱引用
    __weak typeof(self) weakSelf = self;
    [NSURLConnection sendAsynchronousRequest:request queue:[[NSOperationQueue alloc] init] completionHandler:^(NSURLResponse * _Nullable response, NSData * _Nullable data, NSError * _Nullable connectionError) {
        // 出錯則返回
        if (connectionError != nil) {
            NSLog(@"connectionError = %@",connectionError);
            return ;
        }
        // 記錄需要下載的文件總長度
        long long serverFileLength = response.expectedContentLength;
        weakSelf.serverFileLength = serverFileLength;
        // 初始化已下載文件大小
        weakSelf.localFileLength = 0;
        
        NSDictionary *dict = [[NSFileManager defaultManager] attributesOfItemAtPath:weakSelf.filePath error:NULL];
        
        long long  localFileLength = [dict[NSFileSize] longLongValue];
        
        // 如果沒有本地文件,直接下載!
        if (!localFileLength) {
            // 下載新文件
            [weakSelf getFileWithUrlString:urlStr];
        }
        // 如果已下載的大小,大于服務器文件大小,肯定出錯了,刪除文件并從新下載
        if (localFileLength > serverFileLength) {
            // 刪除文件 remove
            [[NSFileManager defaultManager]  removeItemAtPath:self.filePath error:NULL];
            
            // 下載新文件
            [self getFileWithUrlString:urlStr];
            
        } else if (localFileLength == serverFileLength) {
            NSLog(@"文件已經下載完畢");
        } else if (localFileLength && localFileLength < serverFileLength) {
            // 文件下載了一半,則使用斷點續傳
            self.localFileLength = localFileLength;
            
            [self getFileWithUrlString:urlStr WithStartSize:localFileLength endSize:serverFileLength-1];
        }
    }];
    
}
/// 重新開始下載文件(代理方法監聽下載過程)
-(void)getFileWithUrlString:(NSString *)urlString
{
    NSURL *url = [NSURL URLWithString:urlString];
    // 默認就是 GET 請求
    NSURLRequest *request = [NSURLRequest requestWithURL:url];
    
    // 開始請求,并設置代理為self
    NSURLConnection *conn = [[NSURLConnection alloc] initWithRequest:request delegate:self];
    
    // 設置代理的執行線程,一般不在主線程
    [conn setDelegateQueue:[[NSOperationQueue alloc] init]];
    
    // NSUrlConnection 的代理方法是一個特殊的事件源!
    // 開啟運行循環!
    CFRunLoopRun();
}

/* 斷點續傳(代理方法監聽下載過程)
 * startSize:本次斷點續傳開始的位置
 * endSize:本地斷點續傳結束的位置
 */
-(void)getFileWithUrlString:(NSString *)urlString WithStartSize:(long long)startSize endSize:(long long)endSize
{
    // 1. 創建請求!
    NSURL *url = [NSURL URLWithString:urlString];
    // 默認就是 GET 請求
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
    
    // 設置斷點續傳信息
    NSString *range = [NSString stringWithFormat:@"Bytes=%lld-%lld",startSize,endSize];
    [request setValue:range forHTTPHeaderField:@"Range"];
    
    // NSUrlConnection 下載過程!
    NSURLConnection *conn = [[NSURLConnection alloc] initWithRequest:request delegate:self];
    
    // 設置代理的執行線程// 傳一個非主隊列!
    [conn setDelegateQueue:[[NSOperationQueue alloc] init]];
    
    // NSUrlConnection 的代理方法是一個特殊的事件源!
    // 開啟運行循環!
    CFRunLoopRun();
}

#pragma mark - NSURLConnectionDataDelegate
/// 1. 接收到服務器響應
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
    // 服務器響應
    // response.expectedContentLength的大小是要下載文件的大小,而不是文件的總大小。比如請求頭設置了Range[requestM setValue:rangeStr forHTTPHeaderField:@"Range"];則返回的大小為range范圍內的大小
}
/// 2. 接收到數據
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {
    //data當前接收到的網絡數據;
    // 如果這個文件不存在,響應的文件句柄就不會創建!
    NSFileHandle *fileHandle = [NSFileHandle fileHandleForWritingAtPath:self.filePath];
    // 在這個路徑下已經有文件了!
    if (fileHandle) {
        // 將文件句柄移動到文件的末尾
        [fileHandle seekToEndOfFile];
        // 寫入文件的意思(會將data寫入到文件句柄所操縱的文件!)
        [fileHandle writeData:data];
        
        [fileHandle closeFile];
        
    } else {
        // 第一次調用這個方法的時候,在本地還沒有文件路徑(沒有這個文件)!
        [data writeToFile:self.filePath atomically:YES];
    }
}
/// 3. 接收完成
- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
    NSLog(@"下載完成");
}
/// 4. 網絡錯誤
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
    NSLog(@"下載錯誤 %@", error);
}
@end

四、NSData 數據的反序列化

服務器返回的NSData數據類型,事先已經知道,因此可以直接返序列化為實際的類型,例如:json(字典或數組)、.plist(字典或數組)text、xml等:

  • 加載數據
// 加載本地
NSString *path = [[NSBundle mainBundle] pathForResource:@"文件名 " ofType:nil];
NSData *jsonData = [NSData dataWithContentsOfFile:path];

// 從網絡直接加載
NSURL *url = [[NSBundle mainBundle] URLForResource:@"topic_news.json" withExtension:nil];
NSData *data = [NSData dataWithContentsOfURL:url];

  • .plist反序列化

這種很少使用,只是蘋果自己的格式

// 關于選型參數
// NSPropertyListImmutable = 0,不可變  
// NSPropertyListMutableContainers = 1 << 0,容器可變
// NSPropertyListMutableContainersAndLeaves = 1 << 1,容器和葉子可變
// 通常后續直接做字典轉模型,不需要關心是否可變,所以一般直接賦值為0
id result = [NSPropertyListSerialization propertyListWithData:data options:0 format:NULL error:NULL];
  • 關于json數據

    • json數據的本質是字符串,根格式只有兩種可能:數組或字典。

      • 反序列化:從服務器接收到的二進制數據 轉換成 字典或者數組
      // 假設json數據位:[{鍵值對},{鍵值對}],則可以將數據轉化為字典或數組
      

    NSArray *dictArray = [NSJSONSerialization JSONObjectWithData:jsonData options:NSJSONReadingMutableContainers error:nil];
    ```

      - 序列化:將字典或者數組 轉換成 二進制數據,準備發送給服務器  
    
      ```objc
    

// obj為字典或數組
NSData *data = [NSJSONSerialization dataWithJSONObject:obj options:0 error:NULL];
// 轉化前可以判斷是否可以序列化:
(BOOL)isValidJSONObject:(id)obj;
```
* 一個對象能夠被轉換成 JSON 必須符合以下條件:
- 頂級節點,必須是一個 NSArray or NSDictionary
- 所有的對象必須是 NSString, NSNumber, NSArray, NSDictionary, or NSNull
- 所有字典的 key 都必須是 NSString
- NSNumber 不能為空或者無窮大

注意事項:

  • 請求的緩存策略使用 NSURLRequestReloadIgnoringCacheData,忽略本地緩存
  • 服務器響應結束后,要記錄 Etag,服務器內容和本地緩存對比是否變化的重要依據!
  • 在發送請求時,設置 If-None-Match,并且傳入 etag
  • 連接結束后,要判斷響應頭的狀態碼,如果是 304,說明本地緩存內容沒有發生變化,此時可以使用本地緩存來加載數據,如果緩存文件被意外刪除,程序依然運行但會報錯,加載數據也失?。?/li>
NSCachedURLResponse *cachedURLRes = [[NSURLCache sharedURLCache] cachedResponseForRequest:requestM];//cachedURLRes為nil
  • GET 緩存的數據會保存在 Cache 目錄中 /bundleId 下,Cache.db 中

iOS9訪問http網頁

  1. 在Info.plist中添加NSAppTransportSecurity類型Dictionary;
  2. 在NSAppTransportSecurity下添加NSAllowsArbitraryLoads類型Boolean,值設為YES

參考:http://www.cnblogs.com/mddblog/p/5134783.html

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,431評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,637評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,555評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,900評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,629評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,976評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,976評論 3 448
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,139評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,686評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,411評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,641評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,129評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,820評論 3 350
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,233評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,567評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,362評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,604評論 2 380

推薦閱讀更多精彩內容

  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,828評論 18 139
  • 閱讀目錄 一、整體介紹 二、使用的一般步驟 三、舉例說明 四、NSData 數據的反序列化 注意事項: iOS9訪...
    九洲仙人閱讀 805評論 0 0
  • iOS開發系列--網絡開發 概覽 大部分應用程序都或多或少會牽扯到網絡開發,例如說新浪微博、微信等,這些應用本身可...
    lichengjin閱讀 3,702評論 2 7
  • AFHTTPRequestOperationManager 網絡傳輸協議UDP、TCP、Http、Socket、X...
    Carden閱讀 4,375評論 0 12
  • 最近又開始上演小鮮肉仙俠劇了,追劇的時候怎能沒有啤酒和雞腳呀! 外面賣的那又白又胖的泡雞腳,你要是想好好的多活幾年...
    pauline布丁媽閱讀 662評論 4 7