面试题相关

4F3504214246255452FCA18F03FCA708.jpg
  1. 参考 http://www.jianshu.com/p/5559bc15490d

2.ARC的规则就是只要对象没有强指针引用,就会被释放掉,换而言之 只要还有一个强引用指针变量指向对象,那么这个对象就会存在内存中。弱指针指向的对象,会被自动变成空指针(nil指针),从而不会引发野指针错误。

操作完成后手动 设置 某一方为nil 打破循环引用

/*
    //暂停操作不能立刻停止当前任务,只能停止后面准备执行的任务
    //暂停是可以恢复的,YES的时候表示暂停,NO表示恢复
    if (self.queue.suspended) {
        self.queue.suspended = NO;
    }else
    {
         self.queue.suspended = YES;
    }
     */
    //取消,相当于调用了队列中所有操作的取消方法
    //是不可以恢复的,只能取消尚未执行的任务,当前处于执行状态的任务是不能取消的
    [self.queue cancelAllOperations];

(1).获取本地已下载文件的大小
(2).创建下载任务 并设置请求头(设置需要下载的数据位置)
    1.NSURLConnection
       NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
 /*
     表示头500个字节:Range: bytes=0-499
     表示第二个500字节:Range: bytes=500-999
     表示最后500个字节:Range: bytes=-500
     表示500字节以后的范围:Range: bytes=500-
     */
    
    //设置请求头,可以控制下载文件的某一个部分
    NSString *range = [NSString stringWithFormat:@"bytes=%zd-",self.currentSize];
    [request setValue:range forHTTPHeaderField:@"Range"];
 NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:NO];
    
    开始发送网络请求
    [connect start];
    self.connect = connect;

/当接收到服务器响应的时候调用
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
    //创建文件句柄
   
}

//当接收到服务器返回给我们的数据的时候调用
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
    //指向文件末尾
    //写入数据
}

//当请求结束之后调用
- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
    NSLog(@"下载完毕---%@",self.fullPath);
    
    [self.handle closeFile];
    self.handle = nil;
}

输出流也能达到同样的效果

//相关问题补充

1. 苹果为了方便我们拿到数据以后显示或者刷新UI,默认代理方法在主线程中调用,我们可以通过对象方法setDelegateQueue来设置代理执行的队列。
2. 请求数据的过程也可能非常耗时,我们能否将请求数据的操作也放在子线程中进行呢?答案是可以的但是需要注意,initWithRequest会将方法会将NSURLConnection对象加入当前对应的RunLoop中,当我们在子线程中进行网络请求,默认子线程的RunLoop不会自动创建,NSURLConnection对象会被释放,因此我们需要开启子线程中的RunLoop,保证NSURLConnection对象不会被释放。另外,当在子线程中设置请求手动开启调用start方法,就不需要开启子线程RunLoop了,因为start方法内部如果发现RunLoop不存在就会自动创建。
-(void)createNewThread1
{
    [[[NSOperationQueue alloc]init] addOperationWithBlock:^{
        
        NSLog(@"发送请求---%@",[NSThread currentThread]);
        
        //1.url
        NSURL *url = [NSURL URLWithString:@"XXXXX"];
        
        //2.创建请求对象
        NSURLRequest *request = [NSURLRequest requestWithURL:url];
        
        //3.设置代理
        //   NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self];
        //该行代码内部:把connect作为一个source,添加到当前线程的runloop,但是当前线程是子线程,子线程对应的runloop默认是没有创建的,所以添加不成功
        NSURLConnection *connect = [NSURLConnection connectionWithRequest:request delegate:self];
        
        //设置代理方法在哪个线程中调用
        NSOperationQueue *queue  = [[NSOperationQueue alloc]init];
        [connect setDelegateQueue:queue];
        
        [[NSRunLoop currentRunLoop]run];
    }];
    
}


-(void)createNewThread2
{
     [[[NSOperationQueue alloc]init]addOperationWithBlock:^{
         
         NSLog(@"子线程发送请求%@",[NSThread currentThread]);
        //1.url
        NSURL *url = [NSURL URLWithString:@"http://XXXXX"];
        
        //2.创建请求对象
        NSURLRequest *request = [NSURLRequest requestWithURL:url];
        
        //3.设置代理
        //   NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self];
         //如果第三个参数为YES
         //该行代码内部:把connect作为一个source,添加到当前线程的runloop,但是当前线程是子线程,子线程对应的runloop默认是没有创建的,所以添加不成功
//        NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:YES];
         
         //如果第三个参数为NO
         NSURLConnection *connect = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:NO];
        
        //设置代理方法在哪个线程中调用
        NSOperationQueue *queue  = [[NSOperationQueue alloc]init];
        [connect setDelegateQueue:queue];
         
         //如果发现当前线程对应的runloop不存在,那么会自动创建一个runloop
        [connect start];
         
//          [[NSRunLoop currentRunLoop]run];
     }];
    
}

(2)NSURLSessionDataTask

     NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
     NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
 NSString *range = [NSString stringWithFormat:@"bytes=%zd-",self.currentSize];
    [request setValue:range forHTTPHeaderField:@"Range"];
    NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request];
    
    //启动task
    [dataTask resume];
    self.dataTask = dataTask;

//暂停
 [self.dataTask suspend];
//恢复
 [self.dataTask resume];

    /*1.当接收到服务器响应的时候调用*/
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler
{
//创建输出流
    NSOutputStream *stream = [[NSOutputStream alloc]initToFileAtPath:fullPath append:YES];
    self.stream = stream;
    [self.stream open];
    
    NSLog(@"didReceiveResponse");
    //通过该block告诉系统要如何处理服务器返回给我们的数据
    /*
     NSURLSessionResponseCancel = 0, //取消,不接受数据
     NSURLSessionResponseAllow = 1, //接收
     NSURLSessionResponseBecomeDownload = 2,  //变成下载请求
     NSURLSessionResponseBecomeStream //变成stream
     */
    completionHandler(NSURLSessionResponseAllow);
}

/*2.当接收到服务器返回的数据的时候调用,可能被调用多次*/
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
{
    [self.stream write:data.bytes maxLength:data.length];
    
//    NSLog(@"didReceiveData");
    //计算文件下载进度
    self.currentSize +=data.length;
    NSLog(@"%f",1.0 * self.currentSize/self.totalSize);
    self.progressView.progress = 1.0 * self.currentSize/self.totalSize;
}

/*3.当请求结束的时候调用,如果请求失败,那么error有值*/
-(void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
    NSLog(@"didCompleteWithError");
    [self.stream close];
    self.stream = nil;
    
    
}

你可能感兴趣的:(面试题相关)