iOS多线程之3.NSOperation、NSOperationQueue

  GCD是C语言编写的API,对于习惯了面向对象的我们来说可能用起来不习惯,所以Apple又为我们提供了另一种多线程解决方案——使用NSOperation(操作,相当于GCD中的任务)、NSOperationQueue(队列)实现多线程。

1. 使用NSOperation子类封装操作

   NSOperation是一个抽象类,并不能封装操作。我们想封装操作有两种方式:
(1)使用系统提供的NSOperation的两个子类NSInvocationOperation和NSBlockOperation。
(2)自定义继承于NSOperation的子类,实现- (void)main方法。

- (void)viewDidLoad {
    [super viewDidLoad];
     
    //  利用NSInvocationOperation封装操作
    //  可以利用object传递参数
    NSInvocationOperation *invocationOperation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(executeTask1:) object:@{@"key":@"value"}];
    [invocationOperation start];
    
    //  利用NSBlockOperation封装操作
    NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"执行操作2----%@",[NSThread currentThread]);
    }];
   
 [blockOperation start];
}

- (void)executeTask1:(id)object
{
    NSLog(@"执行操作1----%@",[NSThread currentThread]);
}

日志

操作1参数----{
    key = value;
}
执行操作1----<NSThread: 0x282403100>{number = 1, name = main}
执行操作2----<NSThread: 0x282403100>{number = 1, name = main}

  可以看到,NSOperation的子类如果只是封装单个操作,操作都是在当前线程执行的,并没有开启新的线程。
  下面看一下自定义NSOperation子类的使用,以下载图片为例。新建一个CustomOperation类,继承于NSOperation。
  CustomOperation.h

#import <Foundation/Foundation.h>
#import <UIKit/UIKit.h>

@interface CustomOperation : NSOperation

// 传入URL
@property (nonatomic, copy) NSString *strURL;
// 下载完成图片的回调
@property (nonatomic, copy) void (^downLoadImageBLock)(UIImage *image);

@end

  CustomOperation.m

#import "CustomOperation.h"

@implementation CustomOperation

// 1.实现main方法
- (void)main {
    // 2.新建一个自动释放池
    @autoreleasepool {
        // 3.正确响应取消事件
        if (self.isCancelled)  return;
        
        NSURL *url = [NSURL URLWithString:self.strURL];
        NSData *data = [NSData dataWithContentsOfURL:url];
        if (self.isCancelled) 
        {
            url = nil;
            data = nil;
        }
        
        UIImage *image = [UIImage imageWithData:data];
        if (self.isCancelled)  
        {
            image = nil;
            return;
        }
        // 4.在主线程执行回调
        dispatch_async(dispatch_get_main_queue(), ^{
             //5. 用block传值
            if (self.downLoadImageBLock) {
                self.downLoadImageBLock(image);
            }
        });
    }
}

  下面来看看如何使用自定义的NSOperation子类。

// 点击屏幕下载图片
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {

    // 1.创建一个CustomOperation对象
    CustomOperation *operation = [[CustomOperation alloc] init];
    // 2.为URL赋值
    operation1.strURL = @"http://h.hiphotos.baidu.com/zhidao/pic/item/6d81800a19d8bc3ed69473cb848ba61ea8d34516.jpg";
    // 3.处理block传回来的结果
    operation1.DownLoadImageBLock = ^(UIImage *image) {
        self.imageView1.image = image;
    };

   // 开始执行操作
   [operation start];
}

2. 使用NSOperationQueue实现多线程并发

  利用 NSOperationQueue和NSOperation配合使用实现多线程并发的效果只需要两步:
(1)利用NSOperation的子类封装操作。
(2)把操作放入队列中(NSOperationQueue对象)中。

- (void)viewDidLoad
{
    [super viewDidLoad];
 
    // 创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    
    // 封装操作
    NSBlockOperation *operation1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"操作1开始执行");
        // 模拟耗时操作
        sleep(3);
        NSLog(@"执行操作1------%@",[NSThread currentThread]);
    }];
    
    NSBlockOperation *operation2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"操作2开始执行");
        sleep(1);
        NSLog(@"执行操作2------%@",[NSThread currentThread]);
    }];

    NSBlockOperation *operation3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"操作3开始执行");
        sleep(7);
        NSLog(@"执行操作3------%@",[NSThread currentThread]);
    }];

    // 把操作放入队列中
    [queue addOperation:operation1];
    [queue addOperation:operation2];
    [queue addOperation:operation3];
}

日志

操作1开始执行
操作2开始执行
操作3开始执行
执行操作2------<NSThread: 0x282f28000>{number = 3, name = (null)}
执行操作1------<NSThread: 0x282f15640>{number = 4, name = (null)}
执行操作3------<NSThread: 0x282f28500>{number = 5, name = (null)}

  本次在队列中添加了3个操作,系统新创建了3个线程,3个操作并发执行。需要注意的是并不是在队列中添加几个操作,系统就创建几个线程,这是由CPU的使用情况决定。另外,队列里的操作都不会在当前线程中执行

3.使用maxConcurrentOperationCount控制并发数量和实现串行

  另外我们可以决定使用maxConcurrentOperationCount并发执行的操作个数。`queue.maxConcurrentOperationCount = 2;就是两个操作同时开始执行。

- (void)viewDidLoad
{
    [super viewDidLoad];
 
    // 创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    
    //  设置最大并发数
    queue.maxConcurrentOperationCount = 1;
    
    // 封装操作
    NSBlockOperation *operation1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"操作1开始执行");
        // 模拟耗时操作
        sleep(3);
        NSLog(@"执行操作1------%@",[NSThread currentThread]);
    }];
    
    NSBlockOperation *operation2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"操作2开始执行");
        sleep(1);
        NSLog(@"执行操作2------%@",[NSThread currentThread]);
    }];

    NSBlockOperation *operation3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"操作3开始执行");
        sleep(7);
        NSLog(@"执行操作3------%@",[NSThread currentThread]);
    }];

    // 把操作放入队列中
    [queue addOperation:operation1];
    [queue addOperation:operation2];
    [queue addOperation:operation3];  
}

日志

操作2开始执行
操作1开始执行
执行操作2------<NSThread: 0x282902440>{number = 3, name = (null)}
操作3开始执行
执行操作1------<NSThread: 0x282902480>{number = 4, name = (null)}
执行操作3------<NSThread: 0x282905f80>{number = 5, name = (null)}

  注意
(1)设置并发执行的个数必须在操作放入队列中- (void)addOperation:(NSOperation *)op;前面,否则不起效果。
(2)我们并不是设置并发执行的个数为多少,就会真的有多少个操作并发执行,这只是一个期望值,如果设置的值比系统默认最大并发数大,就会以系统默认最大并发数为准。例如系统默认最大并发数为5,你设置了100,最后这个maxConcurrentOperationCount仍然是5。
(3)这个属性maxConcurrentOperationCount默认值-1,意思不做限制。
  实现串行只要设置queue.maxConcurrentOperationCount = 1就可以了,操作就会按着FIFO的原则顺序执行。
日志

操作1开始执行
执行操作1------<NSThread: 0x2815d1280>{number = 3, name = (null)}
操作2开始执行
执行操作2------<NSThread: 0x2815d1280>{number = 3, name = (null)}
操作3开始执行
执行操作3------<NSThread: 0x2815d1280>{number = 3, name = (null)}

  这里需要注意的是虽然是串行,但是并不表示他们会在同一个线程中执行,这次只是例外, 可以试试在队列里添加10000个操作。

4.控制操作的执行顺序

  如果我们想一个操作执行完了,另一个操作才能继续执行,可以添加依赖。

- (void)viewDidLoad
{
    [super viewDidLoad];
 
    // 创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    
    // 封装操作
    NSBlockOperation *operation1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"操作1开始执行");
        // 模拟耗时操作
        sleep(3);
        NSLog(@"执行操作1------%@",[NSThread currentThread]);
    }];

    NSBlockOperation *operation2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"操作2开始执行");
        sleep(1);
        NSLog(@"执行操作2------%@",[NSThread currentThread]);
    }];

    NSBlockOperation *operation3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"操作3开始执行");
        sleep(7);
        NSLog(@"执行操作3------%@",[NSThread currentThread]);
    }];
    
    // 为operation2添加依赖operation3,  表示operation3执行完了,operation2才能执行
    [operation2 addDependency:operation3];
    [operation1 addDependency:operation2];

    // 把操作放入队列中
    [queue addOperation:operation1];
    [queue addOperation:operation2];
    [queue addOperation:operation3];
}

日志

操作3开始执行
执行操作3------<NSThread: 0x280e23e40>{number = 3, name = (null)}
操作2开始执行
执行操作2------<NSThread: 0x280e23e40>{number = 3, name = (null)}
操作1开始执行
执行操作1------<NSThread: 0x280e23e40>{number = 3, name = (null)}

5.NSOperation的其他用法

5.1 利用NSBlockOperation添加多个操作

  可以利用NSBlockOperation的- (void)addExecutionBlock:(void (^)(void))block;方法添加多个操作。

- (void)viewDidLoad {
    [super viewDidLoad];
 
    NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
        sleep(10);
        NSLog(@"执行任务1----%@", [NSThread currentThread]);
    }];
    
    // 添加新操作
    [operation addExecutionBlock:^{
        sleep(8);
        NSLog(@"执行任务2----%@",[NSThread currentThread]);
    }];
    
    [operation addExecutionBlock:^{
        sleep(7);
        NSLog(@"执行任务3----%@",[NSThread currentThread]);
    }];
    
    [operation start];
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    NSLog(@"用户点击了");
}

日志

执行任务2----<NSThread: 0x283494580>{number = 3, name = (null)}
执行任务1----<NSThread: 0x2834ce680>{number = 1, name = main}
执行任务3----<NSThread: 0x283494580>{number = 3, name = (null)}
用户点击了

  这里需要注意的是NSOperation会把部分操作放到当前线程里执行。如果当前线程是主线程并且是耗时操作,会堵塞主线程,影响用户交互。

5.2 执行完的回调

   NSOperation还可以实现CGD group-notify的功能。

// 当operation里所有的操作执行完了执行block里面的代码
    operation.completionBlock = ^{
        NSLog(@"所有的操作都执行完了");
    };
5.3 设置优先级

   可以使用@property NSOperationQueuePriority queuePriority;设置NSOperation的优先级,默认是NSOperationQueuePriorityNormal`。优先级一共有五个值:

typedef NS_ENUM(NSInteger, NSOperationQueuePriority) {
    NSOperationQueuePriorityVeryLow = -8L,// 优先级很低
    NSOperationQueuePriorityLow = -4L,// 优先级低
    NSOperationQueuePriorityNormal = 0,// 默认
    NSOperationQueuePriorityHigh = 4,// 优先级高
    NSOperationQueuePriorityVeryHigh = 8// 优先级很高
};
5.4 操作的取消、执行、完成

NSOperation的三个属性cancelled、executing、finished,分别就是取消,执行,完成。这三个属性都是只读的,我们通过这三个属性可以判断NSOperation的状态,是否取消了,是否正在执行,是否已经完成了。

6.NSOperationQueue的其他用法

1)暂停:
[queue setSuspended:YES];
2)恢复:
[queue setSuspended:NO];
3)判读队列的当前状态:
@property (getter=isSuspended) BOOL suspended;
  当你把队列暂停时,队列里的操作就不执行了。当你滑动列表时,可以先把队列(队列里执行下载图片的操作,列表里的cell上有图片)暂停,不滑动时再恢复,增加APP的流畅性。
4)取消所有操作
- (void)cancelAllOperations;
  队列里的所有操作都不执行了。
  以上就是关于NSOperation的常用操作,用这些基本上就能够满足我们的需求。如果还满足不了怎么办,自定义NSOperation,下一篇文章讲自定义NSOperation。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容