多线程编程-NSOperation

多线程编程-NSOperation

本文目录前言一、NSInvocationOperation

二、NSBlockOperation

三、NSOperation的其他用法四、自定义NSOperation回到顶部前言

1.上一讲简单介绍了NSThread的使用,虽然也可以实现多线程编程,但是需要我们去管理线程的生命周期,还要考虑线程同步、加锁问题,造成一些性能上的开销。我们也可以配合使用NSOperation和NSOperationQueue实现多线程编程,实现步骤大致是这样的:

1> 先将需要执行的操作封装到一个NSOperation对象中

2> 然后将NSOperation对象添加到NSOperationQueue中

3> 系统会自动将NSOperation中封装的操作放到一条新线程中执行在此过程中,我们根本不用考虑线程的生命周期、同步、加锁等问题下面列举一个应用场景,比如微博的粉丝列表:每一行的头像肯定要从新浪服务器下载图片后才能显示的,而且是需要异步下载。

这时候你就可以把每一行的图片下载操作封装到一个NSOperation对象中,上面有6行,所以要创建6个NSOperation对象,然后添加到NSOperationQueue中,分别下载不同的图片,下载完毕后,回到对应的行将图片显示出来。 

2.默认情况下,NSOperation并不具备封装操作的能力,必须使用它的子类,使用NSOperation子类的方式有3种:

1> NSInvocationOperation

2> NSBlockOperation

3> 自定义子类继承NSOperation,实现内部相应的方法这讲先介绍如何用NSOperation封装一个操作,后面再结合NSOperationQueue来使用。

 回到顶部一、NSInvocationOperation

1 NSInvocationOperation *operation = [[[NSInvocationOperation alloc] initWithTarget:self selector:@selector(run:) object:@"mj"] autorelease];2 [operation start];

* 第1行初始化了一个NSInvocationOperation对象,它是基于一个对象和selector来创建操作

* 第2行调用了start方法,紧接着会马上执行封装好的操作,也就是会调用self的run:方法,并且将@"mj"作为方法参数* 这里要注意:默认情况下,调用了start方法后并不会开一条新线程去执行操作,而是在当前线程同步执行操作。

只有将operation放到一个NSOperationQueue中,才会异步执行操作。 

回到顶部二、NSBlockOperation

1.同步执行一个操作1 NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^(){

2        NSLog(@"执行了一个新的操作");

3 }];

4  // 开始执行任务

5 [operation start];

* 第1行初始化了一个NSBlockOperation对象,它是用一个Block来封装需要执行的操作

* 第2行调用了start方法,紧接着会马上执行Block中的内容

* 这里还是在当前线程同步执行操作,并没有异步执行 

2.并发执行多个操作复制代码 

1 NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^(){ 

2NSLog(@"执行第1次操作,线程:%@", [NSThread currentThread]); 

3 }]; 

5 [operation addExecutionBlock:^() { 

6NSLog(@"又执行了1个新的操作,线程:%@", [NSThread currentThread]);

 7 }]; 

 9 [operation addExecutionBlock:^() {

10 NSLog(@"又执行了1个新的操作,线程:%@", [NSThread currentThread]);11 }];

13 [operation addExecutionBlock:^() {

14NSLog(@"又执行了1个新的操作,线程:%@", [NSThread currentThread]);

15 }];

17 // 开始执行任务

18 [operation start];复制代码* 第1行初始化了一个NSBlockOperation对象* 分别在第5、9、13行通过addExecutionBlock:方法添加了新的操作,包括第1行的操作,一共封装了4个操作* 在第18行调用start方法后,就会并发地执行这4个操作,也就是会在不同线程中执行1 

2013-02-02 21:38:46.102 thread[4602:c07] 又执行了1个新的操作,线程:{name = (null), num = 1}

2 2013-02-02 21:38:46.102 thread[4602:3f03] 又执行了1个新的操作,线程:{name = (null), num = 5}

3 2013-02-02 21:38:46.102 thread[4602:1b03] 执行第1次操作,线程:{name = (null), num = 3}

4 2013-02-02 21:38:46.102 thread[4602:1303] 又执行了1个新的操作,线程:{name = (null), num = 4}可以看出,每个操作所在线程的num值都不一样,说明是不同线程 回到顶部三、NSOperation的其他用法

1.取消操作operation开始执行之后, 默认会一直执行操作直到完成,我们也可以调用cancel方法中途取消操作[operation cancel]; 

2.在操作完成后做一些事情如果想在一个NSOperation执行完毕后做一些事情,就调用NSOperation的setCompletionBlock方法来设置想做的事情operation.completionBlock = ^() {    NSLog(@"执行完毕");};

当operation封装的操作执行完毕后,就会回调Block里面的内容 回到顶部四、自定义NSOperation如果NSInvocationOperation和NSBlockOperation不能满足需求,我们可以直接新建子类继承NSOperation,并添加任何需要执行的操作。如果只是简单地自定义NSOperation,只需要重载-(void)main这个方法,在这个方法里面添加需要执行的操作。下面写个子类DownloadOperation来下载图片

1.继承NSOperation,重写main方法DownloadOperation.h复制代码

#import

@protocol DownloadOperationDelegate;

@interface DownloadOperation : NSOperation

// 图片的url路径

@property (nonatomic, copy) NSString *imageUrl;

// 代理@property (nonatomic, assign) iddelegate;

- (id)initWithUrl:(NSString *)url delegate:(id)delegate;@end

// 图片下载的协议

@protocol DownloadOperationDelegate

- (void)downloadFinishWithImage:(UIImage *)image;

@end复制代码DownloadOperation.m复制代码 

1 #import "DownloadOperation.h"

 3 @implementation DownloadOperation 

4 @synthesize delegate = _delegate; 

5 @synthesize imageUrl = _imageUrl; 

6 // 初始化

8 - (id)initWithUrl:(NSString *)url delegate:(id)delegate {

9    if (self = [super init]) {

10        self.imageUrl = url;

11        self.delegate = delegate;

12    }

13    return self;

14 }

15 // 释放内存

16 - (void)dealloc {

17    [super dealloc];

18    [_imageUrl release];

19 }

21 // 执行主任务

22 - (void)main {

23    // 新建一个自动释放池,如果是异步执行操作,那么将无法访问到主线程的自动释放池

24    @autoreleasepool {

25        // ....

26    }

27 }

28 @end

复制代码

* 在第22行重载了main方法,等会就把下载图片的代码写到这个方法中

* 如果这个DownloadOperation是在异步线程中执行操作,也就是说main方法在异步线程调用,那么将无法访问主线程的自动释放池,所以在第24行创建了一个属于当前线程的自动释放池

2.正确响应取消事件

* 默认情况下,一个NSOperation开始执行之后,会一直执行任务到结束,就比如上面的DownloadOperation,默认会执行完main方法中的所有代码。

* NSOperation提供了一个cancel方法,可以取消当前的操作。

* 如果是自定义NSOperation的话,需要手动处理这个取消事件。比如,一旦调用了cancel方法,应该马上终止main方法的执行,并及时回收一些资源。

* 处理取消事件的具体做法是:在main方法中定期地调用isCancelled方法检测操作是否已经被取消,也就是说是否调用了cancel方法,如果返回YES,表示已取消,则立即让main方法返回。

* 以下地方可能需要调用isCancelled方法:

在执行任何实际的工作之前,也就是在main方法的开头。因为取消可能发生在任何时候,甚至在operation执行之前。

执行了一段耗时的操作之后也需要检测操作是否已经被取消

复制代码

1 - (void)main {

2    // 新建一个自动释放池,如果是异步执行操作,那么将无法访问到主线程的自动释放池

3    @autoreleasepool {

4        if (self.isCancelled) return;

5

6        // 获取图片数据

7        NSURL *url = [NSURL URLWithString:self.imageUrl];

8        NSData *imageData = [NSData dataWithContentsOfURL:url];

9

10        if (self.isCancelled) {

11            url = nil;

12            imageData = nil;

13            return;

14        }

15

16        // 初始化图片

17        UIImage *image = [UIImage imageWithData:imageData];

18

19        if (self.isCancelled) {

20            image = nil;

21            return;

22        }

23

24        if ([self.delegate respondsToSelector:@selector(downloadFinishWithImage:)]) {

25            // 把图片数据传回到主线程

26            [(NSObject *)self.delegate performSelectorOnMainThread:@selector(downloadFinishWithImage:) withObject:image waitUntilDone:NO];

27        }

28    }

29 }

复制代码

* 在第4行main方法的开头就先判断operation有没有被取消。如果被取消了,那就没有必要往下执行了

* 经过第8行下载图片后,在第10行也需要判断操作有没有被取消

* 总之,执行了一段比较耗时的操作之后,都需要判断操作有没有被取消

* 图片下载完毕后,在第26行将图片数据传递给了代理(delegate)对象

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

推荐阅读更多精彩内容