Ios 多线程之NSOperation与NSOprationQueue

694 阅读2分钟

在说NSOperation之前,先说一下gcd,gcd 技术是一个轻量的,底层实现隐藏的神奇技术,我们能够通过gcd和block轻松实现多线程编程,有时候,gcd相比其他系统提供的多线程方法更加有效,当然,有时候gcd不是最佳选择,另一个多线程编程的技术 NSOprationQueue 让我们能够将后台线程以队列方式依序执行,并提供更多操作的入口,这和 gcd 的实现有些类似。

这种类似不是一个巧合,在早期,MacOX 与 iOS 的程序都普遍采用Operation Queue来进行编写后台线程代码,而之后出现的gcd技术大体是依照前者的原则来实现的,而随着gcd的普及,在iOS 4 与 MacOS X 10.6以后,Operation Queue的底层实现都是用gcd来实现的。

所以,目前可以利用Operation Queue上层的封装,比较简易的实现更简单的多线程操作。

在复用控件,或者多任务执行的情况下,避免不了要开启多个线程和中断线程。

此时,我们就可以使用NSOperation来异步执行任务和中断任务。 包括IOS UITableView和UICollectionView中的cell复用状态下的多线程操作


@property (strong, nonatomic) NSOperationQueue *operationQueue;
@property (strong, nonatomic) NSMutableDictionary *operationDict;



- (NSMutableDictionary *)operationDict {
  if (!_operationDict) {
    _operationDict = [NSMutableDictionary dictionary];
  }
  return _operationDict;
}

- (NSOperationQueue *)operationQueue {
  if (!_operationQueue) {
    _operationQueue = [[NSOperationQueue alloc] init];
    _operationQueue.maxConcurrentOperationCount = 6;
  }
  return _operationQueue;
}

//控件显示开启任务
- (void)collectionView:(UICollectionView *)collectionView willDisplayCell:(UICollectionViewCell *)cell forItemAtIndexPath:(NSIndexPath *)indexPath {

  if (![self.operationDict objectForKey:@(indexPath.item).stringValue]) {
    NSBlockOperation *operation = [self operationEvent:indexPath.item];
    [self.operationQueue addOperation:operation];
    [self.operationDict setObject:operation forKey:@(indexPath.item).stringValue];
  }
}

//控件消失中断任务
- (void)collectionView:(UICollectionView *)collectionView didEndDisplayingCell:(UICollectionViewCell *)cell forItemAtIndexPath:(NSIndexPath *)indexPath {

  if (self.operationDict[@(indexPath.item).stringValue]) {
    NSBlockOperation *operation = self.operationDict[@(indexPath.item).stringValue];
    [operation cancel];
    [self.operationDict removeObjectForKey:@(indexPath.item).stringValue];
  }
}

//异步任务
- (NSBlockOperation *)operationEvent:(NSInteger)index {
  WEAKSELF
  NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
  
  }];
  return operation;
}

- (void)dealloc
{
  NSLog(@"释放%@",self);
  [self.operationQueue cancelAllOperations];
}

在隐藏和显示复用控件中中断和开启任务。可以在当前控件下处理各种复杂任务而不会冲突。例如图片加载,图片压缩,下载回调,异步读取资源等多种情况下都非常实用。