八. NSOperation的基本使用

一. NSOperation简介

  1. NSOperation是在GCD之后推出的, 以操作队列为两大核心的多线程应用类, 可以说他是对GCD的拓展并且进行了一层面向对象的包装

  2. 根据苹果的解释, NSOperation本身是一个抽象类, 它本身并没有开启线程, 执行任务等能力而是使用它的两大子类:

    • NSBlockOperation: 将操作封装到Block中, 在线程执行该操作时, 执行Block中的代码
    • NSInvocationOperation: 将操作封装到方法中, 在线程执行操作时, 执行指定的方法
    • NSOperation也可以通过自建子类, 来执行任务, 不过由于他的两大子类和GCD可以完成大部分任务, 因此自定义的NSOperation类比较少用
  3. 通过操作队列, 即NSOperationNSOperationQueue结合使用, 即可实现多线程并发执行任务

  4. NSOperation的操作, 需要手动开启, 并且NSOperation的对象是single-shot objec, 即一次性的对象, 当他启用之后, 就不能再次使用了.

  5. NSInvocationOperation的简单使用

    • 创建一个NSInvocationOperation对象, 并且在初始化的时候, 指定该操作要调用的方法

    • 实现调用的方法

    • 开启操作

        - (void)invocation {
            NSInvocationOperation *op1 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download) object:nil];
            [op1 start];
        }
        
        - (void)download {
            // 任务默认是在主线程中执行的
            NSLog(@"download---%@", [NSThread currentThread]);
        }
      
  6. NSBlockOperation的简单使用

    • 创建NSBlockOperation对象, 并且在创建的同时, 要将操作封装到block的内部

    • 当操作开始执行的时候, 系统就会调用block内封装的代码

    • NSBlockOperation对象, 在开启前可以追加任务

    • 当操作中的任务数量大于一的时候, NSBlockOperation就会开启子线程来执行任务

    • 开启操作

        NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
            NSLog(@"op2 --- %@", [NSThread currentThread]);
        }];
        
        // 2. 追加任务,当一个操作中的任务数量大于1的时候,就会开启子线程执行任务
        [op2 addExecutionBlock:^{
            NSLog(@"op2Add --- %@", [NSThread currentThread]);
        }];
        
        [op2 addExecutionBlock:^{
            NSLog(@"op2ADD --- %@", [NSThread currentThread]);
        }];
        
        // 3. 开启任务
        [op2 start];
      
  7. 自定义NSOperation子类

    • 自定义的NSOperation子类, 需要创建一个继承自NSOperation的类

    • 在类中, 实现-main方法, 该方法中的代码, 就是操作开启时要执行的代码

    • 开启操作

        #import "MYOperation.h"
        
        @implementation MYOperation
        
        - (void)main {
            NSLog(@"main --- %@", [NSThread currentThread]);
        }
        
        @end
        
        // 3. 自建Operation类
        - (void)myOP {
            MYOperation *op = [[MYOperation alloc] init];
            
            [op start];
        }
      

二. NSOperationQueue的使用

  1. NSOperationQueue, 即操作队列, 与GCD的队列有一些类似, 他可以通过和NSOperation操作组合, 达到多线程并发的效果

  2. NSOperationQueue的两种队列

    • 主队列: [NSOpedationQueue mainQueue], 凡是在主队列中执行的操作, 都在主线程执行
    • 非主队列: 直接通过[[NSoperationQueue alloc] init]获得的队列, 非主队列同时具备了并发串行的功能
    • 当前队列: [NSOperationQueue currentQueue], 这个方法可以获得, 调用方法时所在的队列, 但并不是创建队列
  3. NSInvocationOperation使用队列

    • 主队列的任务都在主线程执行, 因此不做讨论

    • 非主队列中的任务分为串行和并发, 默认是并发执行的

    • 先创建队列, 然后将任务添加到队列中去

    • 注意: 将任务添加到队列中, 不必手动开启任务, 进入队列的任务会自动执行

        - (void)invocation {
            
            // 1. 封装操作
            NSInvocationOperation *op = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download) object:nil];
            NSInvocationOperation *op2 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download) object:nil];
            NSInvocationOperation *op3 = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(download) object:nil];
            
            // 2. 创建操作队列
        //    NSOperationQueue *queue = [NSOperationQueue mainQueue]; // 主队列,串行队列
        //    NSOperationQueue *queue = [NSOperationQueue currentQueue]; // 当前队列
            NSOperationQueue *queue = [[NSOperationQueue alloc] init]; // 并发队列
            
            // 3. 将操作添加到队列中
            [queue addOperation:op];
            [queue addOperation:op2];
            [queue addOperation:op3];
        }
      
  4. NSBlockOperation使用队列

    • 与上述InvocationOperation的使用方法大致相同

    • 创建操作队列, 为了让任务并发执行, 使用非主队列

    • 将任务添加到队列中

    • 可以给blockOperation追加任务, 追加的任务同样会在子线程中执行

        - (void)block {
        
            // 1. 封装操作
            NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
                NSLog(@"op ---- %@", [NSThread currentThread]);
            }];
            
            NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
                NSLog(@"op2 ---- %@", [NSThread currentThread]);
            }];
            
            // 2. 创建操作队列
            NSOperationQueue *queue = [[NSOperationQueue alloc] init];
        //    NSOperationQueue *queue = [NSOperationQueue mainQueue];
            
            // 3. 追加任务,追加的任务一定会在子线程中执行
            [op addExecutionBlock:^{
                NSLog(@"opADD ---- %@", [NSThread currentThread]);
            }];
            
            // 4. 将操作添加到队列中
            [queue addOperation:op];
            [queue addOperation:op2];
        }
      
  5. 自定义NSOperation子类使用队列

    • 自定义的NSOperation子类, 需要实现main方法, 来规定该操作执行的任务

    • 其他使用步骤与上述的方法相同

        - (void)myOp {
            
            MYOperation *op = [[MYOperation alloc] init];
            MYOperation *op2 = [[MYOperation alloc] init];
            MYOperation *op3 = [[MYOperation alloc] init];
            
            NSOperationQueue *queue = [[NSOperationQueue alloc] init];
            
            [queue addOperation:op];
            [queue addOperation:op2];
            [queue addOperation:op3];
        }
      
  6. 操作队列的一些使用方法

    1. 设置最大并发数: queue.maxConcurrentOperationCount
      • 通过设置这个属性, 可以控制队列是串行还是并行
      • 如果设置为1, 那么该队列就是串行的
      • 如果设置大于1, 那么队列中的人物就是并发执行的
      • 但是如果小于1, 那么该队列就不会执行任何任务了
    2. 暂停/恢复/取消队列中的任务
      • 暂停/恢复: queue.suspended
        • 该属性接收BOOL值, 如果传入YES, 队列当前就会暂停执行任务
        • 如果传入NO, 队列就会继续执行任务, 因此暂停操作是可以恢复的
        • 注意: 如果临时设置为暂停的话, 那么队列会处理完当前任务, 暂时不执行下个任务
      • 取消任务: [queue cancelAllOperations]
        • 该方法会取消队列中的所有任务

        • 注意: 取消任务, 也是需要等当前任务执行结束之后, 后面的任务都取消执行

        • 取消操作是不可恢复的

        • 苹果官方建议, 每当执行一次耗时操作后, 就手动检查一下当前队列是否为取消状态, 如果是, 就直接return, 有利于提高性能

            -(void)main
            {
                //耗时操作1
                for (int i = 0; i<1000; i++) {
                    NSLog(@"任务1-%d--%@",i,[NSThread currentThread]);
                }
                NSLog(@"+++++++++++++++++++++++++++++++++");
          
                //苹果官方建议,每当执行完一次耗时操作之后,就查看一下当前队列是否为取消状态,如果是,那么就直接退出
                //好处是可以提高程序的性能
                if (self.isCancelled) {
                    return;
                }
          
                //耗时操作2
                for (int i = 0; i<1000; i++) {
                    NSLog(@"任务1-%d--%@",i,[NSThread currentThread]);
                }
          
                NSLog(@"+++++++++++++++++++++++++++++++++");
            }
          
  7. 操作依赖和监听

    1. 操作依赖
      • NSOperationQueue可以给队列中的操作设置依赖关系
      • 如: [op addDependency:op2], 则op必须等op2执行完毕后才能执行
      • 这样可以在控制任务并发执行时的先后顺序
    2. 监听: op.completionBlock
      • 给某一个操作设置监听

      • 当监听到这个操作执行完毕的时候, 就会调用block中的代码

      • 通过与操作依赖连用, 可以监听队列中所有的任务是否完成

          - (void)invocation {
              
              NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
                  NSLog(@"op ---- %@", [NSThread currentThread]);
              }];
              
              NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
                  NSLog(@"op2 ---- %@", [NSThread currentThread]);
              }];
              
              NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{
                  NSLog(@"op3 ---- %@", [NSThread currentThread]);
              }];
              
              // 设置监听,该任务会在所有任务都完成之后调用
              op3.completionBlock = ^{
                  NSLog(@"操作已经完成%@", [NSThread currentThread]);
              };
              
              NSOperationQueue *queue = [[NSOperationQueue alloc] init];
              
              // 设置依赖(优先级),op < op2
              [op addDependency:op2];
              [op2 addDependency:op3];
              
              [queue addOperation:op];
              [queue addOperation:op2];
              [queue addOperation:op3];
          }
        

三. GCD和NSOperation的对比

  1. 不同点

    • GCD是纯C语言的API; 而NSOperation已经包装为了Object-C对象操作, 更加的面向对象
    • GCD的任务都是用Block来封装的, 比较轻量级; 而NSOperation则是直接使用对象, 比GCD较为重量级
    • 如果执行一些简单的线程操作, 就使用GCD
    • 如果需要控制线程中的任务, NSOperation的思路比较清晰
  2. NSOperation的优点

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

推荐阅读更多精彩内容