Results show
GIF is a bit big, direct connection: 7 qnbrb.com1.z0.glb.clouddn.com/download.gi…
Knowledge points
- NSOperationQueue Manages the thread queue
- NSURLSession Indicates the network operation
- FMDB database operation
NSOperationQueue
-
Apple provides a set of multi-threaded solutions, based on a higher level of PACKAGING GCD, fully object-oriented.
-
We use NSOperation in conjunction with NSOperationQueue to implement multi-threading and perform asynchronous tasks.
NSOperation implements multi-threading in three steps: 1. Create operation: Encapsulate the operation to be performed in an NSOperation object. 2. Create a queue: Create an NSOperationQueue object. 3. Add the operation to the queue: Add the NSOperation object to the NSOperationQueue object.
-
NSOperation is an abstract class and cannot be used to encapsulate operations. The two subclasses of NSOperation, NSInvocationOperation and NSBlockOperation, as well as custom subclasses derived from NSOperation. Since we need to control the queue execution according to the state of the NSURLSessionTask, we can only customize NSOperation and listen on the state of the NSURLSessionTask in the general KVO mode within the subclass
NSURLSession
NSURLSession is iOS7 used to replace NSURLConnection and the new interface, used for network related operations, including data requests, upload, download, processing authentication tools, can handle the HTTP protocol used in things. NSURLSession does not work directly, but is done by NSURLSessionTask. NSURLSessionTask has three subclasses: NSURLSessionDataTask, upload NSURLSessionUploadTask, download NSURLSessionDownloadTask, you can initialize it with a block, a delegate, It is also possible to use NSURLSessionDataTask for uploading and downloading. Relational structure
NSURLSessionDataTask and NSURLSessionDownloadTask
-
NSURLSessionDownloadTask
- Advantages: using cancelByProducingResumeData cancel the download, can obtain the current download file to resumeData, download contains how much, total size, etc., and can configure the background download function.
- Disadvantages: The NSURLSessionDownloadTask will store the downloaded binaries in the TMP temporary file (the contents of this folder can be emptied at any time), and then move the file to the specified sandbox file when the download is complete.
-
NSURLSessionDataTask
- Advantages: can directly to write binary data into the permanent storage of sandbox file (NSDocumentDirectory NSLibraryDirectory)
- Disadvantages: You need to customize NSFileHandle and store the downloaded data locally
-
Contrast: By comparing the advantages and disadvantages of the two, NSURLSessionDownloadTask must continuously copy TMP binary files to NSDocumentDirectory or NSLibraryDirectory if it wants to realize self-downloading after the process terminates. And constantly fetching resumeData values, which can cause a huge waste of resources and memory overhead. Use NSURLSessionDataTask to avoid binary file copy, the basic data is stored in the database at the beginning of the download, so as to terminate the process at any time, restart the application, can obtain the resource path, resource size, download state, maintain the state before the termination of the application or automatically restart the download.
-
Based on the comparison, I finally chose to use NSURLSessionDataTask for breakpoint download.
FMDB
No more on FMDB.
Train of thought
- 1. Preparing for download: Add the download task to the download task queue, and store the file name, resource path, and download status (waiting for download) in the database
- 2. Start downloading: obtain the total file size, name, update table data: download status is downloading, and write binary data to sandbox using NSFileHandle
- 3. Pause > update table data: how many downloads are currently made, download status changed to: Pause
- 4. Cancel Download > Database to delete the data and remove the data from the task array
- 5. Download Complete > Delete the data in database and remove the data from the task array
- 6. If the application is killed, restart the application
- 6.1. Check whether there is a download queue
- 6.2. Start automatically for queues in download state
- 6.33. Queues in suspended and other conditions remain in their original condition
Add the download task to the queue
There are two parts: the first addition and the pause or failed restart
- (void)addDownloadQueue:(NSString *)fileUrl {
MHDownloadModel *downloadModel = [self fetchDownloadModelWithFileUrl:fileUrl];
if (downloadModel) {
return; } // If the file exists in the database, it is only a pause or failed state, Restart the download (pause - > is downloading) downloadModel = [[MHFileDatabase shareInstance] queryModelWitFileName: fileUrl. LastPathComponent];if(downloadModel) {/ / update the download file the state of the [[MHFileDatabase shareInstance] updateDownloadStatusWithFileName: fileUrl. LastPathComponent downloadStatus:MHDownloadStatusDownloading]; }else{// Insert data, Record the downloaded file [[MHFileDatabase shareInstance] insertFileWithFileName: fileUrl. LastPathComponent filePath: fileUrl fileTotalSize:0]; downloadModel = [MHDownloadModel new]; downloadModel.filePath = fileUrl; } [self.downloadTasks addObject:downloadModel]; [self startDownLoadWithUrl:fileUrl]; }Copy the code
Start the download task
The download may not be immediately downloaded. The start of the download is controlled by NSOperationQueue. By default, the maximum number of concurrent downloads is 3.
- (void)startDownLoadWithUrl:(NSString *)url {
MHDownloadModel *downloadModel = [self fetchDownloadModelWithFileUrl:url];
downloadModel.downloadStatus = MHDownloadStatusDownloadWait;
NSURLSessionDataTask *task = downloadModel.task;
if (task && task.state == NSURLSessionTaskStateRunning) {
return;
}
__weak typeof(self) weakSelf = self;
MHURLSessionTaskOperation *operation = [MHURLSessionTaskOperation operationWithURLSessionTask:nil sessionBlock:^NSURLSessionTask *{
__strong typeof(weakSelf) strongSelf = weakSelf;
NSLog(@"thread : %@, MHCustomOperation operationWithURLSessionTask", [NSThread currentThread]);
return [strongSelf downloadDataTaskWithUrl:url];
}];
[self.operationQueue addOperation:operation];
}
Copy the code
Encapsulate download request:
1. Set the URL. 2. Send the request
- (NSURLSessionDataTask *)downloadDataTaskWithUrl:(NSString *)url{
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:url]];
NSDictionary *dic = [[NSFileManager defaultManager] attributesOfItemAtPath:[self getFilePathWithUrl:url] error:nil];
MHDownloadModel *downloadModel = [self fetchDownloadModelWithFileUrl:url];
downloadModel.currentSize = [dic[@"NSFileSize"] integerValue];
[request setValue:[NSString stringWithFormat:@"bytes=%zd-",downloadModel.currentSize] forHTTPHeaderField:@"Range"];
NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:self delegateQueue:[NSOperationQueue mainQueue]];
NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request];
NSLog(@"Thread: %@, download -- %s", [NSThread currentThread], __func__);
downloadModel.task = dataTask;
return dataTask;
}
Copy the code
Implement agent NSURLSessionDataDelegate
- Start download: get the size of the target file, create a sandbox file You must call completionHandler (NSURLSessionResponseAllow)
- (void)URLSession:(NSURLSession *)session
dataTask:(NSURLSessionDataTask *)dataTask
didReceiveResponse:(NSURLResponse *)response
completionHandler:(void (^)(NSURLSessionResponseDisposition disposition))completionHandler
Copy the code
- Gets the download progress and writes binary data to the sandbox
- (void)URLSession:(NSURLSession *)session
dataTask:(NSURLSessionDataTask *)dataTask
didReceiveData:(NSData *)data
Copy the code
- Download complete or error
- (void)URLSession:(NSURLSession *)session
task:(NSURLSessionTask *)task
didCompleteWithError:(nullable NSError *)error
Copy the code
Full case: Github