NSOperation

NSOperation

NSOperation简介

  • Operation : 操作的意思.
  • 是OC语言中基于GCD的面向对象的封装.
  • 使用起来比GCD更加简单(面向对象)
  • 提供了一些用GCD不好实现的功能.
  • 苹果推荐使用,使用NSOperation不用关心线程以及线程的生命周期.
GCD和OP的关系图.png

NSOperation类的介绍

  1. NSOperation是个抽象类,无法直接使用.因为方法只有声明没有实现.

    作为父类使用的.约束子类共有的属性和方法.

  2. 子类 :

    • NSInvocationOperation
    • NSBlockOperation
    • 自定义NSOperation

    操作默认是异步的.

  3. 队列 : NSOperationQueue

    队列默认是并发的.

  4. 核心 :

    • GCD的核心 : 将任务添加到队列
    • OP的核心 : 将操作添加到队列

使用步骤

  1. 先将需要执行的操作封装到一个NSOperation对象中.创建NSOperation对象.
  2. 将NSOperation对象添加到NSOperationQueue中.
  3. NSOperationQueue会自动将NSOperation取出来.
  4. 将取出的NSOperation封装的操作自动放到一条对应的新线程中执行.
操作添加到队列.png

NSInvocationOperation

  • 核心 : 将操作添加到队列.

NSInvocationOperation 基本使用演练

- (void)demo:(id)parram
{
    // 查看当前线程
    NSLog(@"%@ %@",parram,[NSThread currentThread]);
}

OP调用start方法

- (void)opDemo1
{
    // 创建操作对象
    NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(demo:) object:@"InvocationOperation"];
    // 调用start方法
    [op start];
}
  • [op start]; 方法,会在当前线程执行 @selector方法.

NSInvocationOperation 将操作添加到队列

- (void)opDemo2
{
    // 操作对象 : OP中的操作对象默认是异步执行
    NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(demo:) object:@"InvocationOperation"];
    // 队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    // 将操作添加到队列
    [queue addOperation:op];
}
  • 操作 : 默认是异步执行.

NSInvocationOperation 验证队列并发性

- (void)opDemo3
{
    // 队列 : 默认是并发的
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];

    // 循环的向队列中添加10个操作
    for (int i = 0; i < 10; i++) {
        // 操作对象 : OP中的操作对象默认是异步执行
        NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(demo:) object:@(i)];
        // 将操作添加到队列
        [queue addOperation:op];
    }
}
  • 执行效果 : 会开启多条线程,不是顺序执行.与GCD中并发队列&异步执行效果一样
  • 队列 : 默认是并发的

NSBlockOperation

NSBlockOperation 基本使用演练

NSBlockOperation 操作添加到队列

- (void)opDemo1
{
    // 队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    // 操作 : 默认是异步的
    NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
        // 查看当前线程
        NSLog(@"%@",[NSThread currentThread]);
    }];
    // 将操作添加到队列
    [queue addOperation:op];

    // 在当前线程执行
//    [op start];
}

NSBlockOperation 验证队列并发性

- (void)opDemo2
{
    // 队列 : 默认是并发
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];

    for (int i = 0; i < 10; i++) {
        // 操作 : 默认是异步的
        NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
            // 查看当前线程
            NSLog(@"%d %@",i,[NSThread currentThread]);
        }];
        // 将操作添加到队列
        [queue addOperation:op];
    }
}

开发建议

  • NSOperationQueue只有一种类型.就是并发队列.
  • 在实际开发时,如果要使用到NSOperationQueue,可以直接定义成全局的队列
/// 定义全局队列
@property (nonatomic,strong) NSOperationQueue *queue;
- (NSOperationQueue *)queue
{
    if (_queue==nil) {
        _queue = [[NSOperationQueue alloc] init];
    }
    return _queue;
}

NSBlockOperation 简写

- (void)opDemo3
{
    [self.queue addOperationWithBlock:^{
        // 查看当前线程
        NSLog(@"%@",[NSThread currentThread]);
    }];
}

线程间通信

- (void)opDemo5
{
    [self.queue addOperationWithBlock:^{
        NSLog(@"努力下载中...%@",[NSThread currentThread]);

        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            NSLog(@"更新UI...%@",[NSThread currentThread]);
        }];
    }];
}

NSOperation与GCD对比

GCD

  • 核心概念 : 将任务(block)添加到队列(串行/并发/主队列),并且指定任务执行的函数(同步/异步).
  • GCD是C语言的API.
  • iOS 4.0 推出的,针对多核处理器的并发技术.
  • 任务封装在block中.
  • 要停止已经加入 队列(queue)任务(block) 需要写复杂的代码.
  • 只能设置队列的优先级.
  • 建立任务间的依赖关系比较复杂.
  • 高级功能 :
    • 一次性 once
    • 延迟操作 after
    • 调度组

NSOperation

  • 核心概念:把操作(异步)添加到队列(并发队列)
  • OC 框架,更加面向对象,是对 GCD 的封装.
  • iOS 2.0 推出的,苹果推出 GCD 之后,对NSOperation的底层全部重写
  • 任务封装在Operation对象中的,为我们提供了更多的选择.操作对象更加方便.
  • 可以取消掉队列中的任务,正在执行的任务除外.
  • 可以设置队列中每一个操作的优先级.
  • 可以跨队列设置操作的依赖关系.
  • 高级功能 :
    • 最大操作并发数(GCD不好做)
    • 继续/暂停/全部取消
    • 跨队列设置操作的依赖关系

NSOperation高级功能演练

  • 最大操作并发数
  • 继续/暂停/取消全部
  • 操作的优先级和监听操作执行完成的回调
  • 操作间依赖关系

队列的最大并发数

  • 队列的一个属性.
  • @property NSInteger maxConcurrentOperationCount;
  • 限制同时执行的任务数.
  • 比如,最大并发数设置成3,CPU就最多准备3个线程同时执行3个任务.
  • 线程可以复用.而且在线程回收的间隙可以及时的准备线程保证并发性.

*** 准备队列**

/// 定义全局队列
@property (nonatomic,strong) NSOperationQueue *queue;
  • 懒加载的时候设置最大并发数
- (NSOperationQueue *)queue
{
    if (_queue==nil) {
        _queue = [[NSOperationQueue alloc] init];

        // 设置最大并发数 : 每次只能调度两个操作执行
        _queue.maxConcurrentOperationCount = 2;
    }
    return _queue;
}
  • 演示最大并发数对队列调度任务的影响
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    [self opDemo];
}
- (void)opDemo
{
    NSLog(@"start");

    for (int i = 0; i < 20; i++) {
        [self.queue addOperationWithBlock:^{

            // 休眠一秒钟,演示最大并发数的效果更好
            [NSThread sleepForTimeInterval:1.0];

            NSLog(@"%d %@",i,[NSThread currentThread]);
        }];
    }
}

执行的结果 : 任务是两个两个的执行.

队列的暂停继续和取消全部

  • 在最大并发数的代码基础上演示队列的暂停继续和取消全部

相关的属性和方法介绍

  • - (BOOL)isSuspended;暂停和继续队列的属性.
  • YES代表暂停队列,NO代表恢复队列.
  • cancelAllOperations : 取消队列中的全部操作.
  • cancel : 取消队列中的单个操作.

1. 队列暂停

#pragma mark - 演示队列的暂停
- (IBAction)stop:(id)sender
{
    // 暂停队列之前判断队列中有无操作
    if (self.queue.operationCount == 0) {
        return;
    }

    // 暂停队列
    self.queue.suspended = YES;
    NSLog(@"暂停 %zd",self.queue.operationCount);
}
  • 将队列挂起之后,队列中的操作就不会被调度,但是正在执行的操作不受影
  • operationCount:操作计数,没有执行和没有执行完的操作,都会计算在操作计数之内

注意 : 如果先暂停队列,再添加操作到队列,队列不会调度添加的操作.所以在暂停队列之前要判断队列中有没有任务.如果没有任务就不暂停队列.

2. 队列继续

- (IBAction)resume:(id)sender
{
    // 队列继续
    self.queue.suspended = NO;
    NSLog(@"继续 %zd",self.queue.operationCount);
}

3. 队列取消全部

- (IBAction)cancelAll:(id)sender
{
    [self.queue cancelAllOperations];
    NSLog(@"取消全部 %zd",self.queue.operationCount);
}
  • 一旦调用的 cancelAllOperations方法,队列中的操作,都会被移除,正在执行的操作除外.
  • 正在执行的操作取消不了,如果要取消,需要自定义队列.

操作优先级和监听操作完成回调

  • 操作的优先级 : qualityOfService
    • 无法决定操作执行的先后顺序的,决定的是操作有更多的机会被队列调度执行.
  • 监听操作完成的回调 : @property (nullable, copy) void (^completionBlock)(void) NS_AVAILABLE(10_6, 4_0);
    • 当操作执行结束之后,就会回调,是在子线程中执行的.
- (void)opDemo
{
    // 操作1
    NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
        for (int i = 0; i < 10; i++) {
            // 查看当前线程
            NSLog(@"op1 %@",[NSThread currentThread]);
        }
    }];

    //设置操作的优先级
    op1.qualityOfService = NSQualityOfServiceUserInteractive;

    // 当操作执行结束之后,就会回调,是在子线程中执行的
    [op1 setCompletionBlock:^{
        // 查看当前线程
        NSLog(@"操作结束了 %@",[NSThread currentThread]);
    }];

    [self.queue addOperation:op1];

    // 操作2
    NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
        for (int i = 0; i < 10; i++) {
            // 查看当前线程
            NSLog(@"op2 %@",[NSThread currentThread]);
        }
    }];
    op2.qualityOfService = NSQualityOfServiceBackground;
    [self.queue addOperation:op2];
}

操作间依赖

操作op2依赖于op1
[op2 addDependency:op1];

4. 操作依赖

需求 : 登陆-->付费-->下载-->通知用户

  • 准备需要执行的操作
#pragma mark - 操作依赖
- (void)dependency
{
    NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"登陆 %@",[NSThread currentThread]);
    }];

    NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"付费 %@",[NSThread currentThread]);
    }];

    NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"下载 %@",[NSThread currentThread]);
    }];

    NSBlockOperation *op4 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"通知用户 %@",[NSThread currentThread]);
    }];
}
  • 建立依赖关系 : 不能循环建立操作间依赖关系.否则,队列不调度操作执行
// 操作2依赖操作1
[op2 addDependency:op1];
[op3 addDependency:op2];
[op4 addDependency:op3];

// 不能循环依赖 : 操作不会被调度
// [op1 addDependency:op4];

// waitUntilFinished : 是否等到指定的操作执行结束再执行后面的代码
[self.queue addOperations:@[op1,op2,op3,op4] waitUntilFinished:NO];

// 验证 waitUntilFinished
NSLog(@"end");
  • 建立依赖关系 : 操作间可以跨队列建立依赖关系
// 操作2依赖操作1
[op2 addDependency:op1];
[op3 addDependency:op2];
[op4 addDependency:op3];

// 不能循环依赖 : 操作不会被调度
// [op1 addDependency:op4];

// waitUntilFinished : 是否等到指定的操作执行结束再执行后面的代码
[self.queue addOperations:@[op1,op2,op3] waitUntilFinished:NO];

// 通知用户的操作在主线程中执行
// 操作可以跨队列依赖
[[NSOperationQueue mainQueue] addOperation:op4];

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

推荐阅读更多精彩内容