博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
iOS 开发--多线程
阅读量:6547 次
发布时间:2019-06-24

本文共 5877 字,大约阅读时间需要 19 分钟。

前面在《》一文中讲完了多线程的NSThread,不难发现这种方式的多线程实现起来非常的复杂,为了简化多线程的开发,iOS提供了GCD来实现多线程。GCD有俩个核心的概念:

队列:队列负责管理开发者提交的任务,GCD队列始终以先进先出的方式来处理任务,但由于任务的执行时间并不相同,因此先处理的任务并不一定先结束。队列既可是串行队列,也可是并发队列,串行队列每次只处理一个任务,必须前一任务完成后,才会执行下一任务;并放队列则可同时处理多个任务,So将会有多个任务并发执行。队列底层会维护一个线程池来处理用户提交的任务,线程池的作用就是执行队列管理的任务。串行队列底层的线程池只要维护一个线程即可,并发队列则想反。

任务:任务则为用户提交给队列的工作单元,这些任务将会提交给队列底层维护的线程池执行,因此这些任务会以多线程的方式执行。

综上所述,不难发现,使用GCD只需俩步即可。

1.创建队列。

2.将任务提交给队列。

接下来我让我们详细的玩一玩这GCD把?

首先咱先创建队列

//获取当前执行代码所在的队列(已废弃)dispatch_get_current_queue(void);/*根据制定优先级、额外的旗标来获取系统的全局并发队列,第一个参数指定dispatch queue的优先级,取值可以是DISPATCH_QUEUE_PRIORITY_HIGHT、DISPATCH_QUEUE_PRIORITY_DEFAULT、DISPATCH_QUEUE_PRIORITY_LOW或DISPATCH_QUEUE_PRIORITY_BACKGROUND(iOS 5.0加入的); 第二个参数,目前只能为0或NULL*/dispatch_get_global_queue(long identifier, unsigned long flags); //获取主线程相关联的串行队列 dispatch_get_main_queue(void) /* 根据指定字符串标签创建队列。第二个参数可控制创建串行队列还是并发队列,如果将第二个参数设置为“DISPATCH_QUEUE_SERIAL”,则为串行 设为“DISPATCH_QUEUE_CONCURRENT”为并发队列。在没有启动ARC机制的情况下,通常这种方式创建的队列需要调用dispatch_release()释放引用计数 */ dispatch_queue_create(const char *label, dispatch_queue_attr_t attr); //获取指定队列的字符串标签,dispatch_queue_t代表一个队列 dispatch_queue_get_label(dispatch_queue_t queue);

根据上面的方法,可以创建如下几种队列

1.获取系统默认的全局并发队列

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

2.获取系统主线程关联的串行队列

dispatch_queue_t queue = dispatch_get_main_queue();

3.创建串行队列

dispatch_queue_t queue = dispatch_queue_create("Bison", DISPATCH_QUEUE_SERIAL);

4.创建并发队列

dispatch_queue_t queue = dispatch_queue_create("Bison", DISPATCH_QUEUE_CONCURRENT);

异步提交任务

iOS提供了如下函数来向队列提交任务。下面这些函数很多都有俩个版本:一个接收代码块作为参数的版本,一个接收函数作为参数的版本;其中接收函数的函数名最多多了_f后缀,而且会多一个参数,用于向函数传入应用程序定义的上下文。

//将代码块以异步方式提交给指定队列,该队列底层的线程池将负责执行该代码块dispatch_async(dispatch_queue_t queue, dispatch_block_t block); //将函数以异步方式提交给指定队列,该队列底层的线程池将负责执行该函数 dispatch_async_f(dispatch_queue_t queue,void *context,dispatch_function_t work); //将代码块以同步方式提交给指定队列,该队列底层的线程池将负责执行该代码块 dispatch_sync(dispatch_queue_t queue, dispatch_block_t block); //将函数以同步方式提交给指定队列,该队列底层的线程池将负责执行该函数 dispatch_sync_f(dispatch_queue_t queue,void *context,dispatch_function_t work); //将代码块以异步方式提交给指定的队列,该队列底层的线程池将负责在when指定的时间点执行该代码块 dispatch_after(dispatch_time_t when,dispatch_queue_t queue,dispatch_block_t block); //将函数以异步方式提交给指定的队列,该队列底层的线程池将负责在when指定的时间点执行该函数 dispatch_after_f(dispatch_time_t when,dispatch_queue_t queue,void *context,dispatch_function_t work); //将代码块以异步方式提交给指定的队列,该队列底层的线程池将多次重复执行该代码块 dispatch_apply(size_t iterations, dispatch_queue_t queue,void (^block)(size_t)); //将函数以异步方式提交给指定的队列,该队列底层的线程池将多次重复执行该函数 dispatch_apply_f(size_t iterations, dispatch_queue_t queue,void *context,void (*work)(void *, size_t)); //将代码块提交给指定队列,该队列底层的线程池控制在应用的某个生命周期内仅执行一次。predicate 是指向dispatch_once_t变量的指针,判断是否已经执行过,runtime中很常用 dispatch_once(dispatch_once_t *predicate, dispatch_block_t block);

下面是简单的几个案例

// 定义2个队列dispatch_queue_t serialQueue;dispatch_queue_t concurrentQueue;- (void)viewDidLoad{    [super viewDidLoad]; // 创建串行队列 serialQueue = dispatch_queue_create("fkjava.queue", DISPATCH_QUEUE_SERIAL); // 创建并发队列 concurrentQueue = dispatch_queue_create("fkjava.queue" , DISPATCH_QUEUE_CONCURRENT); } - (IBAction)serial:(id)sender { // 依次将2个代码块提交给串行队列 // 必须等到第1个代码块完成后,才能执行第2个代码块。 dispatch_async(serialQueue, ^(void) { for (int i = 0 ; i < 100; i ++) { NSLog(@"%@===……-……==%d" , [NSThread currentThread] , i); } }); dispatch_async(serialQueue, ^(void) { for (int i = 0 ; i < 100; i ++) { NSLog(@"%@------%d" , [NSThread currentThread] , i); } }); } - (IBAction)concurrent:(id)sender { // 依次将2个代码块提交给并发队列 // 两个代码块可以并发执行 dispatch_async(concurrentQueue, ^(void) { for (int i = 0 ; i < 100; i ++) { NSLog(@"%@===……-……==%d" , [NSThread currentThread] , i); } }); dispatch_async(concurrentQueue, ^(void) { for (int i = 0 ; i < 100; i ++) { NSLog(@"%@------%d" , [NSThread currentThread] , i); } }); }

编译执行不难看出,第一个是串行队列,第二个是并行队列。

下面我们将简单的举个异步下载图片的例子

代码如下:

// 将代码块提交给系统的全局并发队列dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0),^(void){    NSString* url = @"http://allluckly.cn/images/blog/applepay/6.png"; // 从网络获取数据 NSData *data = [[NSData alloc] initWithContentsOfURL:[NSURL URLWithString:url]]; // 将网络数据初始化为UIImage对象 UIImage *image = [[UIImage alloc]initWithData:data]; if(image != nil) { // 将代码块提交给主线程关联的队列,该代码块将会由主线程完成 dispatch_async(dispatch_get_main_queue(), ^{ self.iv.image = image; }); // ① } else { NSLog(@"---下载图片出现错误---"); } });

这里值的注意的是我们的图片是http的,xcode7以上必须设置下info.plist文件设置下网络,否则无法成功!
不难看出上面的送上面的例子中通过创建全局并发队列,该代码块负责从网络下载图片,下载完成后交给主线程执行。

同步提交任务

dispatch_async()函数则会以同步方式提交代码块,该函数必须等到代码块执行结束才会返回。如果程序使用该函数先后提交了俩个代码块(并发队列),也必须等第一个任务执行完后才会执行第二个任务。如下?

- (IBAction)clicked:(id)sender{    // 以同步方式先后提交2个代码块    dispatch_sync(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0) , ^(void){ for (int i = 0 ; i < 100; i ++) { NSLog(@"%@=====%d" , [NSThread currentThread] , i); [NSThread sleepForTimeInterval:0.1]; } }); // 必须等第一次提交的代码块执行完成后,dispatch_sync()函数才会返回, // 程序才会执行到这里,才能提交第二个代码块。 dispatch_sync(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0) , ^(void){ for (int i = 0 ; i < 100; i ++) { NSLog(@"%@-----%d" , [NSThread currentThread] , i); [NSThread sleepForTimeInterval:0.1]; } }); }

多次执行任务

dispatch_apply()函数将代码块多次重复执行,如果该代码块被提交给并发队列,系统可以使用多个线程并发执行同一个代码块。下面我们简单的看个?.

- (IBAction)clicked:(id)sender{    // 控制代码块执行5次    dispatch_apply(5    , dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0) // time形参代表当前正在执行第几次 , ^(size_t time) { NSLog(@"===执行【%lu】次===%@" , time, [NSThread currentThread]); }); }

只执行一次任务

dispatch_once()将代码块提交给指定队列,该队列底层的线程池控制在应用的某个生命周期内仅执行一次。predicate 是指向dispatch_once_t变量的指针,判断是否已经执行过,runtime中很常用。系统直接用主线程执行该函数提交的代码块。下面我们简单的看个?.

- (IBAction)clicked:(id)sender{        static dispatch_once_t onceToken;    dispatch_once(&onceToken, ^{ NSLog(@"==执行代码块=="); // 线程暂停3秒 [NSThread sleepForTimeInterval:3]; }); }

 

 

文/Bison(简书作者)
原文链接:http://www.jianshu.com/p/2f18ffcd4c22
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。
你可能感兴趣的文章
zabbix从2.2.3升级到最新稳定版3.2.1
查看>>
我有一个网站,想提高点权重
查看>>
浅谈(SQL Server)数据库中系统表的作用
查看>>
微软邮件系统Exchange 2013系列(七)创建发送连接器
查看>>
程序员杂记系列
查看>>
【树莓派】制作树莓派所使用的img镜像(一)
查看>>
理解网站并发量
查看>>
spring整合elasticsearch之环境搭建
查看>>
TensorFlow 架构与设计-编程模型【转】
查看>>
如何运行Struts2官网最新Demo?
查看>>
'ascii' codec can't decode byte 0xe6 in position 0: ordinal not in range(128)
查看>>
XDebug 教程
查看>>
js 去html 标签
查看>>
好久不见
查看>>
小tips:JS中的children和childNodes
查看>>
二叉树的遍历
查看>>
Oracle的FIXED_DATE参数
查看>>
NDK配置
查看>>
(转)@ContextConfiguration注解说明
查看>>
[置顶] ※数据结构※→☆线性表结构(queue)☆============队列 顺序存储结构(queue sequence)(八)...
查看>>