iOS NSOperation学习及总结

这篇文章对iOS多线程技术NSOperation的常用方法做了简单总结
GCD请见这篇
本文代码

NSOperation:

- 简介:

  • 是苹果在GCD的基础上做了一次面向对象的封装
  • 核心概念和GCD很像
  • NSOperation是一个抽象类,想要封装操作,需要使用子类
  • 可以自定义子类,继承NSOperation,实现多线程操作

- NSOperation的子类:

  • NSInvocationOperation:
    NSInvocationOperation *op = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(run) object:nil];
    [op start];

打印结果:

------<NSThread: 0x600000260ac0>{number = 1, name = main}

结论:
如果仅仅是封装了操作,然后调用start方法
他是不会开线程的
就像调用了performSelectorr一样
想要开线程,必须把任务加到队列中去
因为NSOperation底层是GCD,GCD就是把任务加入到队列,根据队列和函数的情况来决定是否开启线程

  • NSBlockOperation:
    NSBlockOperation *op = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"下载1------%@", [NSThread currentThread]);
    }];
    //添加额外任务
    [op addExecutionBlock:^{
        NSLog(@"下载2------%@", [NSThread currentThread]);
    }];
    [op addExecutionBlock:^{
        NSLog(@"下载3------%@", [NSThread currentThread]);
    }];
    [op start];

打印结果:

下载1------<NSThread: 0x60800006f180>{number = 1, name = main}
下载2------<NSThread: 0x6000000791c0>{number = 3, name = (null)}
下载3------<NSThread: 0x608000079e00>{number = 4, name = (null)}

结论:
和NSInvocationOperation一样
如果只是封装了操作并调用了start方法
只是会在主线程执行任务
但是如果通过addExecutionBlock添加了新任务
那么新任务会在子线程执行(这里是比较特殊的地方)

- NSOperationQueue:

NSOperation可以用start方法来执行任务,但是默认是同步的
如果将NSOperation添加到NSOperationQueue中,系统会自动异步执行NSOperation中的操作

    //创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc]init];
    
    //创建操作(NSInvocationOperation)
    NSInvocationOperation *op1 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download1) object:nil];
    NSInvocationOperation *op2 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download2) object:nil];
    
    //创建操作(NSBlockOperation)
    NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"download3 --- %@", [NSThread currentThread]);
    }];
    [op3 addExecutionBlock:^{
        NSLog(@"download4 --- %@", [NSThread currentThread]);
    }];
    [op3 addExecutionBlock:^{
        NSLog(@"download5 --- %@", [NSThread currentThread]);
    }];
    NSBlockOperation *op4 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"download6 --- %@", [NSThread currentThread]);
    }];
    
    [queue addOperation:op1];
    [queue addOperation:op2];
    [queue addOperation:op3];
    [queue addOperation:op4];

打印结果:

download6 --- <NSThread: 0x608000269080>{number = 6, name = (null)}
download2 --- <NSThread: 0x600000265240>{number = 4, name = (null)}
download1 --- <NSThread: 0x600000265200>{number = 3, name = (null)}
download3 --- <NSThread: 0x6000002655c0>{number = 5, name = (null)}
download4 --- <NSThread: 0x608000265f80>{number = 7, name = (null)}
download5 --- <NSThread: 0x600000265300>{number = 8, name = (null)}

结论:
NSOperation有两种队列:

  • 主队列:
[NSOperationQueue mainQueue];

凡是添加到主队列中的任务,都会在主线程执行

  • 非主队列
[[NSOperationQueue alloc]init];

同时包含了串行和并发的功能
就像上面的代码,把任务添加到队列以后,自动在子线程中执行
如果想用非主线程实现串行,则需要设置并发量

- 其他用法:

  • 最大并发数:
// 设置最大并发操作数
    queue.maxConcurrentOperationCount = 2;

设置了最大并发数,就是限制了系统开线程的数量
如果设置为1
任务就会串行执行

  • 挂起:
self.queue.suspended = YES;

当队列中的任务正在执行的时候,设置挂起为YES
任务会暂停
但是没有从内存中销毁
当设置为NO的时候,任务会继续进行

  • 取消所有任务:
[self.queue cancelAllOperations];

调用queue的cancelAllOperations方法
相当于调用了queue中每个NSOperation的cancel方法
所有的任务就被取消了
如果想继续任务
需要重新定制任务加入队列
但是需要注意的是:
如果调用cancel方法的时候,NSOperation有一个任务正在执行
那么需要这个任务执行完了以后,再实现cancel

  • 自定义NSOperation:

继承NSOperation类
可以实现自定义NSOperation
需要执行的任务在main方法中实现

#import "HXOperation.h"
@implementation HXOperation
/**
 * 需要执行的任务
 */
-(void)main
{
    for (NSInteger i = 0; i<1000; i++) {
        NSLog(@"download1 -%zd-- %@", i, [NSThread currentThread]);
    }
    if (self.isCancelled) return;
    for (NSInteger i = 0; i<1000; i++) {
        NSLog(@"download2 -%zd-- %@", i, [NSThread currentThread]);
    }
    if (self.isCancelled) return;
    for (NSInteger i = 0; i<1000; i++) {
        NSLog(@"download3 -%zd-- %@", i, [NSThread currentThread]);
    }
    if (self.isCancelled) return;
}
@end

在外部这么使用就可以:

    // 创建队列
    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    //添加NSOperation
    [queue addOperation:[[HXOperation alloc] init]];

这样就会自动执行main中的任务了
但是苹果建议我们自定义NSOperation的时候,如果内部有耗时操作
那么应该在每一个耗时操作结束以后,检查一下当前任务有没有被取消
因为很有可能NSOperation在执行任务的时候,外部调用了他的cancel方法
如果被取消了就不要再执行剩下的任务了:

if (self.isCancelled) return;
  • 任务依赖和任务监听

我们知道把多个任务加入队列后
系统从队列中把任务取出来并发执行
但是谁先执行取决于CPU先调度那条线程
当我们需要执行某个任务之后再执行其他任务的话(比如下载完图片后再对图片进行处理)
那么就需要设置依赖
或者可以对任务做监听

    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    
    NSBlockOperation *op1 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"download1----%@", [NSThread  currentThread]);
    }];
    NSBlockOperation *op2 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"download2----%@", [NSThread  currentThread]);
    }];
    NSBlockOperation *op3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"download3----%@", [NSThread  currentThread]);
    }];
    NSBlockOperation *op4 = [NSBlockOperation blockOperationWithBlock:^{
        for (NSInteger i = 0; i<10; i++) {
            NSLog(@"download4----%@", [NSThread  currentThread]);
        }
    }];
    NSBlockOperation *op5 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"download5----%@", [NSThread  currentThread]);
    }];
    //监听任务执行完毕后 进行其他操作
    op5.completionBlock = ^{
        NSLog(@"op5执行完毕---%@", [NSThread currentThread]);
    };
    
    // 设置依赖
    [op3 addDependency:op1];
    [op3 addDependency:op2];
    [op3 addDependency:op4];

    [queue addOperation:op1];
    [queue addOperation:op2];
    [queue addOperation:op3];
    [queue addOperation:op4];
    [queue addOperation:op5];
  • 线程之间的通信:

在子线程下载图片后,回到主线程更新UI:

    NSOperationQueue *queue = [[NSOperationQueue alloc] init];
    __block UIImage *image1 = nil;
    // 下载图片1
    NSBlockOperation *download1 = [NSBlockOperation blockOperationWithBlock:^{
        // 图片的网络路径
        NSURL *url = [NSURL URLWithString:@"http://img.pconline.com.cn/images/photoblog/9/9/8/1/9981681/200910/11/1255259355826.jpg"];
        // 加载图片
        NSData *data = [NSData dataWithContentsOfURL:url];
        // 生成图片
        image1 = [UIImage imageWithData:data];
    }];
    __block UIImage *image2 = nil;
    // 下载图片2
    NSBlockOperation *download2 = [NSBlockOperation blockOperationWithBlock:^{
         // 图片的网络路径
        NSURL *url = [NSURL URLWithString:@"http://pic38.nipic.com/20140228/5571398_215900721128_2.jpg"];
        // 加载图片
        NSData *data = [NSData dataWithContentsOfURL:url];
        // 生成图片
        image2 = [UIImage imageWithData:data];
    }];
    // 合成图片
    NSBlockOperation *combine = [NSBlockOperation blockOperationWithBlock:^{
        // 开启新的图形上下文
        UIGraphicsBeginImageContext(CGSizeMake(100, 100));
        // 绘制图片
        [image1 drawInRect:CGRectMake(0, 0, 50, 100)];
        image1 = nil;
        [image2 drawInRect:CGRectMake(50, 0, 50, 100)];
        image2 = nil;
        // 取得上下文中的图片
        UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
        // 结束上下文
        UIGraphicsEndImageContext();
        // 回到主线程显示图片
        [[NSOperationQueue mainQueue] addOperationWithBlock:^{
            self.imageView.image = image;
        }];
    }];
    //设置依赖 保证下载完图片 再去合成
    [combine addDependency:download1];
    [combine addDependency:download2];
    
    [queue addOperation:download1];
    [queue addOperation:download2];
    [queue addOperation:combine];

感谢阅读
你的支持是我写作的唯一动力

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

推荐阅读更多精彩内容

  • NSThread 第一种:通过NSThread的对象方法 NSThread *thread = [[NSThrea...
    攻城狮GG阅读 789评论 0 3
  • 在这篇文章中,我将为你整理一下 iOS 开发中几种多线程方案,以及其使用方法和注意事项。当然也会给出几种多线程的案...
    张战威ican阅读 600评论 0 0
  • 多线程基本概念 单核CPU,同一时间cpu只能处理1个线程,只有1个线程在执行 。多线程同时执行:是CPU快速的在...
    WeiHing阅读 697评论 1 5
  • 作业1. 好好回想一下,小时候呆呆看过什么。挑选一个印象或一幅画面写出来。 小时候呆呆看过什么已经想不起了,不过最...
    狐皮克阅读 169评论 2 0
  • 找一个懂你的人和找一个爱你的人不是一回事。 懂你什么呢,爱你可以是无条件的,懂你不仅不可能,要“懂”也是有条件的。...
    子侠阅读 593评论 1 2