iOS多线程

整理下几种常用的处理多线程的类,并将该类对应的常用方法整理下,采用简单直观的例子展示。

1、NSThread

NSThread的创建方法有三种:

  • init方法,需要start启动
  • detachNewThreadSelector方法,自动启动
  • performSelectorInBackground方法,自动启动

init方法

NSThread *threadInit = [[NSThread alloc] initWithTarget:self selector:@selector(thread_initMethod:) object:@"initMethod"];
[threadInit start];
- (void)thread_initMethod:(NSObject *)object
{
    NSLog(@"init方法 --> object:%@, %@",object, [NSThread currentThread]);
}

输出结果:

init方法 --> object:initMethod, <NSThread: 0x60000027b180>{number = 3, name = (null)}

是在子线程中完成。

detachNewThreadSelector方法

[NSThread detachNewThreadSelector:@selector(thread_detch:) toTarget:self withObject:@"detachNewThread"];
- (void)thread_detch:(NSObject *)object
{
    NSLog(@"detach方法 --> object:%@, %@",object, [NSThread currentThread]);
}

输出结果:

detach方法 --> object:detachNewThread, <NSThread: 0x60000027b0c0>{number = 4, name = (null)}

是在子线程中完成。

performSelectorInBackground方法

[self performSelectorInBackground:@selector(thread_perform:) withObject:@"performSelectorInBackground"];
- (void)thread_perform:(NSObject *)object
{
    NSLog(@"perform方法--> object:%@, %@",object, [NSThread currentThread]);
}

输出结果:

perform方法--> object:performSelectorInBackground, <NSThread: 0x60000007a2c0>{number = 5, name = (null)}

是在子线程中完成。

2、GCD

GCD中的基本概念:

  • 任务:执行的代码块
  • 队列:存放任务的地方
  • 同步/异步:执行的方式

GCD中的三种队列类型:

  • The main queue:与主线程功能相同。实际上,提交至main queue(主队列)的任务会在主线程中执行。main queue可以调用dispatch_get_main_queue()来获得。因为main queue是与主线程相关的,所以这是一个串行队列。
  • Global queues:全局队列是并发队列,并由整个进程共享。进程中存在四个全局队列:高、中(默认)、低、后台四个优先级队列。可以调用dispatch_get_global_queue函数传入优先级来访问队列。
  • 其他队列: 其他队列是用函数dispatch_queue_create创建的队列。创建队列,第一个参数是表示debug用的,Apple建议我们使用倒置域名来命名队列;第二个参数表示队列类型:串行(DISPATCH_QUEUE_SERIAL)或者并发(DISPATCH_QUEUE_CONCURRENT)。

串行同步

dispatch_queue_t serialQueue = dispatch_queue_create("com.multithreading.serialQueue", DISPATCH_QUEUE_SERIAL);
dispatch_sync(serialQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"串行同步:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_sync(serialQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"串行同步:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_sync(serialQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"串行同步:%i --> %@",i, [NSThread currentThread]);
    }
});

输出结果:

串行同步:0 --> <NSThread: 0x600000072d40>{number = 1, name = main}
串行同步:1 --> <NSThread: 0x600000072d40>{number = 1, name = main}
串行同步:2 --> <NSThread: 0x600000072d40>{number = 1, name = main}
串行同步:0 --> <NSThread: 0x60400007ee80>{number = 1, name = main}
串行同步:1 --> <NSThread: 0x60400007ee80>{number = 1, name = main}
串行同步:2 --> <NSThread: 0x60400007ee80>{number = 1, name = main}
串行同步:0 --> <NSThread: 0x60400007ee80>{number = 1, name = main}
串行同步:1 --> <NSThread: 0x60400007ee80>{number = 1, name = main}
串行同步:2 --> <NSThread: 0x60400007ee80>{number = 1, name = main}

都是在主线程中完成,没有开辟新线程。

串行异步

dispatch_queue_t serialQueue = dispatch_queue_create("com.multithreading.serialQueue", DISPATCH_QUEUE_SERIAL);
dispatch_async(serialQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"串行异步:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_async(serialQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"串行异步:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_async(serialQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"串行异步:%i --> %@",i, [NSThread currentThread]);
    }
});

输出结果:

串行异步:0 --> <NSThread: 0x604000467900>{number = 3, name = (null)}
串行异步:1 --> <NSThread: 0x604000467900>{number = 3, name = (null)}
串行异步:2 --> <NSThread: 0x604000467900>{number = 3, name = (null)}
串行异步:0 --> <NSThread: 0x604000467900>{number = 3, name = (null)}
串行异步:1 --> <NSThread: 0x604000467900>{number = 3, name = (null)}
串行异步:2 --> <NSThread: 0x604000467900>{number = 3, name = (null)}
串行异步:0 --> <NSThread: 0x604000467900>{number = 3, name = (null)}
串行异步:1 --> <NSThread: 0x604000467900>{number = 3, name = (null)}
串行异步:2 --> <NSThread: 0x604000467900>{number = 3, name = (null)}

开辟了一个新线程。

并发同步

dispatch_queue_t concurrentQueue = dispatch_queue_create("com.multithreading.concurrent", DISPATCH_QUEUE_CONCURRENT);
dispatch_sync(concurrentQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"并发同步:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_sync(concurrentQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"并发同步:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_sync(concurrentQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"并发同步:%i --> %@",i, [NSThread currentThread]);
    }
});

输出结果:

并发同步:0 --> <NSThread: 0x60000007c980>{number = 1, name = main}
并发同步:1 --> <NSThread: 0x60000007c980>{number = 1, name = main}
并发同步:2 --> <NSThread: 0x60000007c980>{number = 1, name = main}
并发同步:0 --> <NSThread: 0x60000007c980>{number = 1, name = main}
并发同步:1 --> <NSThread: 0x60000007c980>{number = 1, name = main}
并发同步:2 --> <NSThread: 0x60000007c980>{number = 1, name = main}
并发同步:0 --> <NSThread: 0x60000007c980>{number = 1, name = main}
并发同步:1 --> <NSThread: 0x60000007c980>{number = 1, name = main}
并发同步:2 --> <NSThread: 0x60000007c980>{number = 1, name = main}

在主线程中完成,没有开辟新线程。

并发异步

dispatch_queue_t concurrentQueue = dispatch_queue_create("com.multithreading.concurrent", DISPATCH_QUEUE_CONCURRENT);
dispatch_async(concurrentQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"并发异步:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_async(concurrentQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"并发异步:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_async(concurrentQueue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"并发异步:%i --> %@",i, [NSThread currentThread]);
    }
});

输出结果:

并发异步:0 --> <NSThread: 0x600000277480>{number = 4, name = (null)}
并发异步:0 --> <NSThread: 0x604000465f80>{number = 3, name = (null)}
并发异步:0 --> <NSThread: 0x604000466140>{number = 5, name = (null)}
并发异步:1 --> <NSThread: 0x600000277480>{number = 4, name = (null)}
并发异步:1 --> <NSThread: 0x604000465f80>{number = 3, name = (null)}
并发异步:1 --> <NSThread: 0x604000466140>{number = 5, name = (null)}
并发异步:2 --> <NSThread: 0x600000277480>{number = 4, name = (null)}
并发异步:2 --> <NSThread: 0x604000465f80>{number = 3, name = (null)}
并发异步:2 --> <NSThread: 0x604000466140>{number = 5, name = (null)}

本例子中开了三条新线程。

主线程同步

- (void)mainQueueSync
{
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"任务1:%@",[NSThread currentThread]);
    });
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"任务2:%@",[NSThread currentThread]);
    });
    dispatch_sync(dispatch_get_main_queue(), ^{
        NSLog(@"任务3:%@",[NSThread currentThread]);
    });
}

输出结果:

主线程锁死,崩溃

原因:主队列同步会先执行任务1,当前主线程正在执行mainQueueSync方法,造成执行任务1等待mainQueueSync方法结束,mainQueueSync方法等待任务1~3任务结束。

线程通讯

dispatch_async(dispatch_get_global_queue(0, 0), ^{
    NSLog(@"做一些费时的东西");
    dispatch_async(dispatch_get_main_queue(), ^{
        NSLog(@"主线程刷新UI");
    });
});

输出结果:

做一些费时的东西
主线程刷新UI

栅栏

dispatch_queue_t queue = dispatch_queue_create("com.multithreading.fence", DISPATCH_QUEUE_CONCURRENT);
dispatch_async(queue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"任务1:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_async(queue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"任务2:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_barrier_async(queue, ^{
    NSLog(@"------并发异步执行栅栏分割------");
});
dispatch_async(queue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"任务3:%i --> %@",i, [NSThread currentThread]);
    }
});
dispatch_async(queue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"任务4:%i --> %@",i, [NSThread currentThread]);
    }
});

输出结果:

任务2:0 --> <NSThread: 0x604000069940>{number = 4, name = (null)}
任务1:0 --> <NSThread: 0x60000026b780>{number = 3, name = (null)}
任务2:1 --> <NSThread: 0x604000069940>{number = 4, name = (null)}
任务1:1 --> <NSThread: 0x60000026b780>{number = 3, name = (null)}
任务2:2 --> <NSThread: 0x604000069940>{number = 4, name = (null)}
任务1:2 --> <NSThread: 0x60000026b780>{number = 3, name = (null)}
------并发异步执行栅栏分割------
任务4:0 --> <NSThread: 0x604000069940>{number = 4, name = (null)}
任务3:0 --> <NSThread: 0x60000026b780>{number = 3, name = (null)}
任务4:1 --> <NSThread: 0x604000069940>{number = 4, name = (null)}
任务3:1 --> <NSThread: 0x60000026b780>{number = 3, name = (null)}
任务4:2 --> <NSThread: 0x604000069940>{number = 4, name = (null)}
任务3:2 --> <NSThread: 0x60000026b780>{number = 3, name = (null)}

异步是没有顺序的,但是添加了栅栏,可以将任务分为两部分,先做第一部分完毕后,才会做第二部分。

队列组

dispatch_group_t group = dispatch_group_create();
    dispatch_queue_t queue = dispatch_queue_create("com.multithreading.group", DISPATCH_QUEUE_CONCURRENT);
dispatch_group_async(group, queue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"任务1:%i --> %@",i,[NSThread currentThread]);
    }
});
dispatch_group_async(group, queue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"任务2:%i --> %@",i,[NSThread currentThread]);
    }
});
dispatch_group_async(group, queue, ^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"任务3:%i --> %@",i,[NSThread currentThread]);
    }
});
dispatch_group_notify(group, dispatch_get_main_queue(), ^{
    NSLog(@"任务完了,回到主线程刷新UI");
});

输出结果:

任务1:0 --> <NSThread: 0x60400027a080>{number = 3, name = (null)}
任务3:0 --> <NSThread: 0x604000279fc0>{number = 5, name = (null)}
任务2:0 --> <NSThread: 0x60400027a0c0>{number = 4, name = (null)}
任务1:1 --> <NSThread: 0x60400027a080>{number = 3, name = (null)}
任务2:1 --> <NSThread: 0x60400027a0c0>{number = 4, name = (null)}
任务3:1 --> <NSThread: 0x604000279fc0>{number = 5, name = (null)}
任务1:2 --> <NSThread: 0x60400027a080>{number = 3, name = (null)}
任务2:2 --> <NSThread: 0x60400027a0c0>{number = 4, name = (null)}
任务3:2 --> <NSThread: 0x604000279fc0>{number = 5, name = (null)}
任务完了,回到主线程刷新UI

开辟了三条线程,在任务1,2,3完成后会,回调dispatch_group_notify方法,回到主线程刷新UI。

3、NSOperation

NSOperation的创建:

  • 使用NSInvocationOperation子类,需要start开启
  • 使用NSBlockOperation子类,需要start开启
  • 使用继承NSOperation的子类,需要start开启

NSInvocationOperation

NSInvocationOperation *invocationOperation = [[NSInvocationOperation alloc] initWithTarget:self selector:@selector(invocationOperation:) object:@"invocation"];
[invocationOperation start];
- (void)invocationOperation:(NSObject *)object
{
    NSLog(@"invocation方法:%@",[NSThread currentThread]);
}

输出结果:

invocation方法:<NSThread: 0x600000071ec0>{number = 1, name = main}

没有开新线程。

NSBlockOperation

NSBlockOperation *blockOperation = [NSBlockOperation blockOperationWithBlock:^{
    NSLog(@"block方法:%@",[NSThread currentThread]);
}];
[blockOperation addExecutionBlock:^{
    NSLog(@"任务1:%@",[NSThread currentThread]);
}];
[blockOperation addExecutionBlock:^{
    NSLog(@"任务2:%@",[NSThread currentThread]);
}];
[blockOperation addExecutionBlock:^{
    NSLog(@"任务3:%@",[NSThread currentThread]);
}];
[blockOperation start];

输出结果:

block方法:<NSThread: 0x7feaf1701590>{number = 1, name = main}
任务1:<NSThread: 0x7f839bc21280>{number = 2, name = (null)}
任务2:<NSThread: 0x7f839be44180>{number = 3, name = (null)}
任务3:<NSThread: 0x7f839be1f470>{number = 4, name = (null)}

blockOperationWithBlock方法:主线程实现,不加入队列就不会开辟新线程
addExecutionBlock方法:会开新线程

继承NSOperation的子类

此例子中继承NSOperation的子类名为DZROperation ,在.m文件中重写main函数来实现任务

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

@interface DZROperation : NSOperation

@end


// DZROperation.m
#import "DZROperation.h"

@implementation DZROperation

- (void)main
{
    for (int i = 0; i < 3; i++) {
        NSLog(@"DZROperation --> %i, %@",i, [NSThread currentThread]);
    }
}

@end
DZROperation *operation = [[DZROperation alloc] init];
[operation start];

输出结果:

DZROperation --> 0, <NSThread: 0x7fd22ee01ae0>{number = 1, name = main}
DZROperation --> 1, <NSThread: 0x7fd22ee01ae0>{number = 1, name = main}
DZROperation --> 2, <NSThread: 0x7fd22ee01ae0>{number = 1, name = main}

没有加入队列,是在主线程中实现,未开辟新线程。

NSOperationQueue

NSOperationQueue队列类型:

  • 主队列
  • 非主队列(串行,并发)

队列NSOperationQueue有个参数最大并发数:maxConcurrentOperationCount

  • maxConcurrentOperationCount默认-1,直接开启并发,所以非主队列默认是开启并发
  • maxConcurrentOperationCount > 1,进行并发
  • maxConcurrentOperationCount = 1,表示不开线程,是串行
  • maxConcurrentOperationCount系统会限制一个最大值,所以设maxConcurrentOperationCount很大也是无意义的
添加自定义NSOperation类到队列中
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
DZROperation *operation1 = [[DZROperation alloc] init];
DZROperation *operation2 = [[DZROperation alloc] init];
[queue addOperation:operation1];
[queue addOperation:operation2];

输出结果:

DZROperation --> 0, <NSThread: 0x7fc46660c030>{number = 2, name = (null)}
DZROperation --> 0, <NSThread: 0x7fc466605c70>{number = 3, name = (null)}
DZROperation --> 1, <NSThread: 0x7fc46660c030>{number = 2, name = (null)}
DZROperation --> 1, <NSThread: 0x7fc466605c70>{number = 3, name = (null)}
DZROperation --> 2, <NSThread: 0x7fc46660c030>{number = 2, name = (null)}
DZROperation --> 2, <NSThread: 0x7fc466605c70>{number = 3, name = (null)}

此处的NSOperationQueue是默认并发队列,这里开了2个新线程

直接添加任务
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
queue.maxConcurrentOperationCount = 1;
[queue addOperationWithBlock:^{
    NSLog(@"添加任务1:%@",[NSThread currentThread]);
}];
[queue addOperationWithBlock:^{
    NSLog(@"添加任务2:%@",[NSThread currentThread]);
}];
[queue addOperationWithBlock:^{
    NSLog(@"添加任务3:%@",[NSThread currentThread]);
}];

输出结果:

maxConcurrentOperationCount = -1输出:
添加任务2:<NSThread: 0x7fa3a2f169f0>{number = 3, name = (null)}
添加任务1:<NSThread: 0x7fa3a2d09240>{number = 2, name = (null)}
添加任务3:<NSThread: 0x7fa3a2d0a3d0>{number = 4, name = (null)}

maxConcurrentOperationCount = 1输出:
添加任务1:<NSThread: 0x7fe65270b6a0>{number = 2, name = (null)}
添加任务2:<NSThread: 0x7fe65270b6a0>{number = 2, name = (null)}
添加任务3:<NSThread: 0x7fe65270b6a0>{number = 2, name = (null)}

直接在block中实现任务。

operationQueue间通讯
// 主队列
NSOperationQueue *mainOperation = [NSOperationQueue mainQueue];
// 并发队列
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
[queue addOperationWithBlock:^{
    NSLog(@"做复杂操作");
    
    [mainOperation addOperationWithBlock:^{
        NSLog(@"刷新UI");
    }];
}];

输出结果:

做复杂操作
刷新UI

任务的执行先后设定 -- 类似GCD的栅栏
NSOperationQueue *queue = [[NSOperationQueue alloc] init];
NSBlockOperation *operation1 = [NSBlockOperation blockOperationWithBlock:^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"任务1:%d, %@",i, [NSThread currentThread]);
    }
}];
NSBlockOperation *operation2 = [NSBlockOperation blockOperationWithBlock:^{
    for (int i = 0; i < 3; i++) {
        NSLog(@"任务2:%d, %@",i, [NSThread currentThread]);
    }
}];

// 任务1依赖任务2
[operation1 addDependency:operation2];
[queue addOperation:operation1];
[queue addOperation:operation2];

输出结果:

任务2:0, <NSThread: 0x7fc82160c390>{number = 2, name = (null)}
任务2:1, <NSThread: 0x7fc82160c390>{number = 2, name = (null)}
任务2:2, <NSThread: 0x7fc82160c390>{number = 2, name = (null)}
任务1:0, <NSThread: 0x7fc82160c390>{number = 2, name = (null)}
任务1:1, <NSThread: 0x7fc82160c390>{number = 2, name = (null)}
任务1:2, <NSThread: 0x7fc82160c390>{number = 2, name = (null)}

依赖关系,任务1依赖任务2,任务2完成后才能完成任务1。但是不能相互依赖,会造成锁死。

最最后:
详情demo请转接这里

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

推荐阅读更多精彩内容