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;
}
网友评论