NSURLSession的基礎用法

weixin_34234823發表於2017-02-21

NSURLSession的基礎用法

    - (void)viewDidLoad {
    [super viewDidLoad];
   
    NSURLSessionConfiguration *sessionConfig = [NSURLSessionConfiguration defaultSessionConfiguration];
    NSURLSession *urlSession = [NSURLSession sessionWithConfiguration:sessionConfig delegate:self delegateQueue:[[NSOperationQueue alloc] init]]; 
    NSMutableURLRequest *req = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:kRemotePAJsonURL]];
    req.HTTPMethod = @"POST";
    NSURLSessionDataTask *dataTask = [urlSession dataTaskWithRequest:req];
    [dataTask resume];
    NSLog(@"本次dataTask:%@", dataTask);
}
    
    - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask
didReceiveResponse:(NSURLResponse *)response
 completionHandler:(void (^)(NSURLSessionResponseDisposition disposition))completionHandler
{
    NSLog(@"收到響應:%@     \ndataTask:%@", response, dataTask);
    
    self.mData = [NSMutableData data];
    
    NSURLSessionResponseDisposition disposition = NSURLSessionResponseAllow;
    completionHandler(disposition);
}

- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask
    didReceiveData:(NSData *)data
{
    NSLog(@"%@收到data:%ld",[NSThread currentThread] ,data.length);
    
    [self.mData appendData:data];
    
    if (self.mData.length > 3000) {
        [dataTask cancel];
    }
}

- (void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task
didCompleteWithError:(nullable NSError *)error
{
    NSLog(@"完成, error:%@", error);
    if (!error) {
        //在完成的時候,之前收到的data怎麼取到?不借助其他的變數,在該方法裡取不到?
        NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:self.mData options:NSJSONReadingAllowFragments error:nil];
        NSLog(@"%@", dict);
    }
    //不把本次session Invalidate,那麼session持有的delegate不會被釋放.
    [session finishTasksAndInvalidate];
}

對於方法

+ (NSURLSession *)sessionWithConfiguration:(NSURLSessionConfiguration *)configuration 
                                  delegate:(nullable id <NSURLSessionDelegate>)delegate 
                             delegateQueue:(nullable NSOperationQueue *)queue;

delegate和delegateQueue會被urlSession強引用.按蘋果的文件說明:delegate會在URLSession:didBecomeInvalidWithError結束後釋放.但事實上,要想讓delegate在didBecomeInvalidWithError結束後釋放,需要先把session Invalidate.否則,session持有的delegate不會被釋放.這句話的正確理解應該是,當一個session invalidate後,delegate要等到URLSession:didBecomeInvalidWithError結束後才會被釋放.對於delegateQueue,實際使用時delegateQueue不能是主佇列的.當delegateQueue不是主佇列時,didReceiveData:方法將隨機在某個執行緒執行.

基本上一個APP,生成一個urlSession就夠了.沒必要一次請求,建立一個session,請求結束後又將session Invalidate.因此也就沒必要去管delegate和delegateQueue的記憶體釋放問題,這三個物件基本上是等到APP結束才會銷燬的.最佳做法之一就是使用GCD確保只生成一個session,然後使用該session來管理所有的請求.

對於代理方法: - (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition disposition))completionHandler
在該方法中,為什麼收到響應後,還要呼叫completionHandler block?
因為在該方法中,通過disposition引數,呼叫completionHandler後,可以更細粒度的控制本次請求是繼續還是取消還是轉為下載任務.如果是取消,則後面請求的響應體不會接收.如果是轉為下載任務,那麼通過呼叫completionHandler,NSURLSession將呼叫Delegate的 URLSession:dataTask:didBecomeDownloadTask:方法並將新生成的Download task物件作為引數傳入。在此呼叫之後,Delegate將不再接收來自Data task的回撥訊息,並開始接收Download task的回撥訊息。
注意:如果不呼叫

NSURLSessionResponseDisposition disposition = NSURLSessionResponseAllow;
completionHandler(disposition);

後面的didReceiveData:代理方法將不會執行.

疑問

  1. NSURLSession物件是被誰強引用了?如何釋放?
    NSURLSession物件應該是被系統的runloop強引用了,就類似於定時器一樣,需要invalid後,才會被釋放銷燬.
    題外話:如果timer屬性是strong,那麼invalidate後最好將其置為nil,否則invalid後timer因為還有人持有它,而不能銷燬.strong情況下,timer的釋放: [self.timer invalidate];self.timer = nil;定時器物件是註冊到runloop裡的,應該通過invalidate來告訴runloop釋放它.所以self不應該持有該物件,因此timer屬性最好為weak.

  2. NSURLSession的生命週期?

  3. 在didCompleteWithError:完成的時候,之前收到的data怎麼取到?不借助其他的變數,在該方法裡取不到?

相關文章