[OC] NSURLSession

来源:http://www.cnblogs.com/yuxiuyi/archive/2016/12/23/6213728.html
-Advertisement-
Play Games

NSURLConnection在iOS9被宣佈棄用,NSURLSession從13年發展到現在,終於迎來了它獨步江湖的時代.NSURLSession是蘋果在iOS7後為HTTP數據傳輸提供的一系列介面,比NSURLConnection強大,坑少,好用.今天從使用的角度介紹下. ...


有的程式員老了,還沒聽過NSURLSession
有的程式員還嫩,沒用過NSURLConnection
有的程式員很單純,他只知道AFN.

NSURLConnection在iOS9被宣佈棄用,NSURLSession從13年發展到現在,終於迎來了它獨步江湖的時代.NSURLSession是蘋果在iOS7後為HTTP數據傳輸提供的一系列介面,比NSURLConnection強大,坑少,好用.今天從使用的角度介紹下.

除了NSURLSession,文中還會頻繁地出現NSURLSessionConfigurationNSURLSessionTask兩個類.先認識一下,混個臉熟吧.

使用NSURLSession,攏共分兩步:

  • 第一步 通過NSURLSession的實例創建task
  • 第二部 執行task

既然兩步裡面都出現了task,就先說說它吧.
NSURLSessionTask可以簡單理解為任務:如數據請求任務,下載任務,上傳任務and so on.我們使用的是他的子類們:

  • NSURLSessionTask(抽象類)
    • NSURLSessionDataTask
      • NSURLSessionUploadTask
    • NSURLSessionDownloadTask

從這幾個子類的名字就可以大概猜出他們的作用了.接下來我們就從不同類型的任務出發,來使用session.

NSURLSessionDataTask

字面上看是和數據相關的任務,但其實dataTask完全可以勝任downloadTask和uploadTask的工作.這可能也是我們使用最多的task種類.

簡單GET請求

如果請求的數據比較簡單,也不需要對返回的數據做一些複雜的操作.那麼我們可以使用帶block



// 快捷方式獲得session對象
NSURLSession *session = [NSURLSession sharedSession];
NSURL *url = [NSURL URLWithString:@"http://www.daka.com/login?username=daka&pwd=123"];
// 通過URL初始化task,在block內部可以直接對返回的數據進行處理
NSURLSessionTask *task = [session dataTaskWithURL:url
                               completionHandler:^(NSData *data, NSURLResponse *response, NSError error) {
    NSLog(@"%@", [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]);
}];

// 啟動任務
[task resume];

Tips:

  • 所有類型的task都要調用resume方法才會開始進行請求.

簡單POST請求

POST和GET的區別就在於request,所以使用session的POST請求和GET過程是一樣的,區別就在於對request的處理.

NSURL *url = [NSURL URLWithString:@"http://www.daka.com/login"];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
request.HTTPMethod = @"POST";
request.HTTPBody = [@"username=daka&pwd=123" dataUsingEncoding:NSUTF8StringEncoding];

NSURLSession *session = [NSURLSession sharedSession];
// 由於要先對request先行處理,我們通過request初始化task
NSURLSessionTask *task = [session dataTaskWithRequest:request
                                   completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) { NSLog(@"%@", [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]); }];
[task resume];

NSURLSessionDataDelegate代理方法

NSURLSession提供了block方式處理返回數據的簡便方式,但如果想要在接收數據過程中做進一步的處理,仍然可以調用相關的協議方法.NSURLSession的代理方法和NSURLConnection有些類似,都是分為接收響應、接收數據、請求完成幾個階段.

// 使用代理方法需要設置代理,但是session的delegate屬性是只讀的,要想設置代理只能通過這種方式創建session
NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
                                                      delegate:self
                                                 delegateQueue:[[NSOperationQueue alloc] init]];

// 創建任務(因為要使用代理方法,就不需要block方式的初始化了)
NSURLSessionDataTask *task = [session dataTaskWithRequest:[NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.daka.com/login?userName=daka&pwd=123"]]];

// 啟動任務
[task resume];

//對應的代理方法如下:

// 1.接收到伺服器的響應
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler {
    // 允許處理伺服器的響應,才會繼續接收伺服器返回的數據
    completionHandler(NSURLSessionResponseAllow);
}

// 2.接收到伺服器的數據(可能調用多次)
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data {
    // 處理每次接收的數據
}

// 3.請求成功或者失敗(如果失敗,error有值)
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
    // 請求完成,成功或者失敗的處理
}

 

Tips:

關鍵點在代碼註釋裡面都有提及,重要的地方再強調一下:

  • 如果要使用代理方法,需要設置代理,但從NSURLSession的頭文件發現session的delegate屬性是只讀的.因此設置代理要通過session的初始化方法賦值:sessionWithConfiguration:delegate:delegateQueue:其中:
    • configuration參數(文章開始提到的)需要傳遞一個配置,我們暫且使用預設的配置[NSURLSessionConfiguration defaultSessionConfiguration]就好(後面會說下這個配置是幹嘛用的);
    • delegateQueue參數表示協議方法將會在哪個隊列(NSOperationQueue)裡面執行.
  • NSURLSession在接收到響應的時候要先對響應做允許處理:completionHandler(NSURLSessionResponseAllow);,才會繼續接收伺服器返回的數據,進入後面的代理方法.值得一提的是,如果在接收響應的時候需要對返回的參數進行處理(如獲取響應頭信息等),那麼這些處理應該放在前面允許操作的前面.

NSURLSessionDownloadTask

文件下載可以使用NSURLSessionDownloadTask這個子類.

簡單下載

NSURLSessionDownloadTask同樣提供了通過NSURL和NSURLRequest兩種方式來初始化並通過block進行回調的方法.下麵以NSURL初始化為例:

NSURLSession *session = [NSURLSession sharedSession];
NSURL *url = [NSURL URLWithString:@"http://www.daka.com/resources/image/icon.png"] ;
NSURLSessionDownloadTask *task = [session downloadTaskWithURL:url completionHandler:^(NSURL *location, NSURLResponse *response, NSError *error) {
    // location是沙盒中tmp文件夾下的一個臨時url,文件下載後會存到這個位置,由於tmp中的文件隨時可能被刪除,所以我們需要自己需要把下載的文件挪到需要的地方
    NSString *path = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:response.suggestedFilename];
    // 剪切文件
    [[NSFileManager defaultManager] moveItemAtURL:location toURL:[NSURL fileURLWithPath:path] error:nil];
}];
    // 啟動任務
    [task resume];

 

Tips:

  • 需要註意的就是需要將下載到tmp文件夾的文件轉移到需要的目錄.原因在代碼中已經貼出.
  • response.suggestedFilename是從相應中取出文件在伺服器上存儲路徑的最後部分,如數據在伺服器的url為http://www.daka.com/resources/image/icon.png, 那麼其suggestedFilename就是icon.png.

NSURLSessionDownloadDelegate代理方法

同樣的,downloadTask也提供了配套的代理方法

// 每次寫入調用(會調用多次)
- (void)URLSession:(NSURLSession *)session downloadTask:(NSURLSessionDownloadTask *)downloadTask didWriteData:(int64_t)bytesWritten totalBytesWritten:(int64_t)totalBytesWritten totalBytesExpectedToWrite:(int64_t)totalBytesExpectedToWrite {
// 可在這裡通過已寫入的長度和總長度算出下載進度
CGFloat progress = 1.0 * totalBytesWritten / totalBytesExpectedToWrite; NSLog(@"%f",progress);
}

// 下載完成調用
- (void)URLSession:(NSURLSession *)session
      downloadTask:(NSURLSessionDownloadTask *)downloadTask
      didFinishDownloadingToURL:(NSURL *)location {
    // location還是一個臨時路徑,需要自己挪到需要的路徑(caches下麵)
    NSString *filePath = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:downloadTask.response.suggestedFilename];
    [[NSFileManager defaultManager] moveItemAtURL:location toURL:[NSURL fileURLWithPath:filePath] error:nil];
}

// 任務完成調用
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {

}

 

NSURLSessionUploadTask

在NSURLSession中,文件上傳方式主要有以下兩種:

NSURLSessionUploadTask *task =[[NSURLSession sharedSession] uploadTaskWithRequest:request
                                           fromFile:fileName
                                  completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {}];
和
[self.session uploadTaskWithRequest:request
                            fromData:body
                   completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
 NSLog(@"-------%@", [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:nil]);
 }];

 

處於安全性考慮,通常我們會使用POST方式進行文件上傳,所以較多使用第二種方式.

但是,NSURLSession並沒有為我們提供比NSURLConnection更方便的文件上傳方式.方法中body處的參數需要填寫request的請求體(http協議規定格式的大長串).因為你有90%的可能性用了AFNetworking,即使是自己寫的應該也是copy,所以代碼就不貼了我們只說方法呵呵噠.

斷點下載

NSURLSessionDownloadTask提供了與斷點下載相關的幾個方法:

// 使用這種方式取消下載可以得到將來用來恢復的數據,保存起來
[self.task cancelByProducingResumeData:^(NSData *resumeData) {
    self.resumeData = resumeData;
}];

// 由於下載失敗導致的下載中斷會進入此協議方法,也可以得到用來恢復的數據
- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
    // 保存恢複數據
    self.resumeData = error.userInfo[NSURLSessionDownloadTaskResumeData];
}

// 恢復下載時接過保存的恢複數據
self.task = [self.session downloadTaskWithResumeData:self.resumeData];
// 啟動任務
[self.task resume];

 

以目前我對NSURLSession的理解這種斷點下載只支持應用內斷點,如果程式在下載過程中途關閉,則不能恢復下載.(暫時對NSURLSession理解還不全面,不敢妄下斷論,如有不妥簡友們可以溝通下)

其他

此外,task們自身有都擁有下麵幾個方法

- (void)suspend;
- (void)resume;
- (void)cancel;

suspend可以讓當前的任務暫停
resume方法不僅可以啟動任務,還可以喚醒suspend狀態的任務
cancel方法可以取消當前的任務,你也可以向處於suspend狀態的任務發送cancel消息,任務如果被取消便不能再恢復到之前的狀態.

NSURLSessionConfiguration

簡單地說,就是session的配置信息.如:

NSURLSessionConfiguration *config = [NSURLSessionConfiguration defaultSessionConfiguration];
// 超時時間
config.timeoutIntervalForRequest = 10;
// 是否允許使用蜂窩網路(後臺傳輸不適用)
config.allowsCellularAccess = YES;
// 還有很多可以設置的屬性

有沒有發現我們使用的Configuration都是預設配置:[NSURLSessionConfiguration defaultSessionConfiguration],其實它的配置有三種類型:

+ (NSURLSessionConfiguration *)defaultSessionConfiguration;
+ (NSURLSessionConfiguration *)ephemeralSessionConfiguration;
+ (NSURLSessionConfiguration *)backgroundSessionConfigurationWithIdentifier:(NSString *)identifier

表示了NSURLSession幾種不同的工作模式.
預設的配置會將緩存存儲在磁碟上,第二種瞬時會話模式不會創建持久性存儲的緩存,第三種後臺會話模式允許程式在後臺進行上傳下載工作.

除了支持任務的暫停和斷點續傳,我覺得NSURLSession之於NSURLConnection的最偉大的進步就是支持後臺上傳下載任務,這又是一個可以深入討論的話題.但在這方面我還沒有進行深入的研究,待後續瞭解之後另行開貼.

PS:AFNetWorking從2.0版本就有了基於NSURLSession的系列封裝,感興趣的童鞋自行前往瞭解.

 

文/CoderAO(簡書作者)
原文鏈接:http://www.jianshu.com/p/fafc67475c73
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 對象屬性: 1、private修飾的屬性:只能在本類內部訪問,分類和外部都不能訪問(徹底私有) 2、fileprivate修飾的屬性:在本類和分類中可以訪問,外部不能訪問(部分私有) 3、直接用let或var修飾的屬性:在本類、分類和外部都可以訪問(開放) 4、在分類中只能聲明計算屬性,不能聲明存儲 ...
  • 在Android3.0之後,Google對UI導航設計上進行了一系列的改革,其中有一個非常好用的新功能就是引入的ActionBar,他用於取代3.0之前的標題欄,並提供更為豐富的導航效果。ActionBar的主要目的是: 1.提供一個用於識別應用程式的標示和用戶的位置的專用空間。 2.在不同的應用程 ...
  • 一直不懂別人口中說的原生開發、混合式開發。今天突然看了一篇文章講解的是什麼叫做原生App?移動 Web App?混合APP?分享給大家。 原生App是專門針對某一類移動設備而生的,它們都是直接安裝到設備里,而用戶一般也是通過網路商店或者賣場來獲取例如 The App Store與Android Ap ...
  • 1、到github網址上下載zip壓縮包https://github.com/danielgindi/Charts 2、然後將解壓後的文件夾整個拖到自己的工程文件夾下(很多教程只讓拖xcodeproj文件,但我拖了不好使),如下圖: 3、然後用xcode打開工程,將剛纔拖進去的Charts-mast ...
  • access_token分兩種 access_token就是一種許可權集合的含義. ...
  • 直接看代碼吧!!! ...
  • 現實中:電臺要發佈消息,通過廣播把消息廣播出去,使用收音機,就可以收聽廣播,得知這條消息。Android中:系統在運行過程中,會產生許多事件,那麼某些事件產生時,比如:電量改變、收發簡訊、撥打電話、屏幕解鎖、開機,系統會發送廣播。 只要應用程式接收到這條廣播,就知道系統發生了相應的事件,從而執行相應 ...
  • 一、網路保存數據介紹 可以使用網路來保存數據,在需要的時候從網路上獲取數據,進而顯示在App中。 用網路保存數據的方法有很多種,對於不同的網路數據採用不同的上傳與獲取方法。 本文利用LeanCloud來進行網路數據的存儲。 LeanCloud是一種簡單高效的數據和文件存儲服務。感興趣的可以查看網址: ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...