欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

iOS应用程序中通过dispatch队列控制线程执行的方法

程序员文章站 2024-02-10 12:35:16
gcd编程的核心就是dispatch队列,dispatch block的执行最终都会放进某个队列中去进行,它类似nsoperationqueue但更复杂也更强大,并且可以嵌...

gcd编程的核心就是dispatch队列,dispatch block的执行最终都会放进某个队列中去进行,它类似nsoperationqueue但更复杂也更强大,并且可以嵌套使用。所以说,结合block实现的gcd,把函数闭包(closure)的特性发挥得淋漓尽致。

dispatch队列的生成可以有这几种方式:

1. dispatch_queue_t queue = dispatch_queue_create("com.dispatch.serial", dispatch_queue_serial); //生成一个串行队列,队列中的block按照先进先出(fifo)的顺序去执行,实际上为单线程执行。第一个参数是队列的名称,在调试程序时会非常有用,所有尽量不要重名了。

2. dispatch_queue_t queue = dispatch_queue_create("com.dispatch.concurrent", dispatch_queue_concurrent); //生成一个并发执行队列,block被分发到多个线程去执行

3. dispatch_queue_t queue = dispatch_get_global_queue(dispatch_queue_priority_default, 0); //获得程序进程缺省产生的并发队列,可设定优先级来选择高、中、低三个优先级队列。由于是系统默认生成的,所以无法调用dispatch_resume()和dispatch_suspend()来控制执行继续或中断。需要注意的是,三个队列不代表三个线程,可能会有更多的线程。并发队列可以根据实际情况来自动产生合理的线程数,也可理解为dispatch队列实现了一个线程池的管理,对于程序逻辑是透明的。

官网文档解释说共有三个并发队列,但实际还有一个更低优先级的队列,设置优先级为dispatch_queue_priority_background。xcode调试时可以观察到正在使用的各个dispatch队列。

4. dispatch_queue_t queue = dispatch_get_main_queue(); //获得主线程的dispatch队列,实际是一个串行队列。同样无法控制主线程dispatch队列的执行继续或中断。

线程操作示例
为了方便地使用gcd,苹果提供了一些方法方便我们将block放在主线程或后台线程执行,或者延后执行。使用的例子如下所示:

复制代码 代码如下:

 //  后台执行:
 dispatch_async(dispatch_get_global_queue(0, 0), ^{
      // something
 });

复制代码 代码如下:

 // 主线程执行:
 dispatch_async(dispatch_get_main_queue(), ^{
      // something
 });

复制代码 代码如下:

 // 一次性执行:
 static dispatch_once_t oncetoken;
 dispatch_once(&oncetoken, ^{
     // code to be executed once
 });

复制代码 代码如下:

 // 延迟2秒执行:
 double delayinseconds = 2.0;
 dispatch_time_t poptime = dispatch_time(dispatch_time_now, delayinseconds * nsec_per_sec);
 dispatch_after(poptime, dispatch_get_main_queue(), ^(void){
     // code to be executed on the main queue after delay
 });

dispatch_queue_t也可以自己定义,如要自定义queue,可以用dispatch_queue_create方法,示例如下:
复制代码 代码如下:

 // 自定义dispatch_queue_t
 dispatch_queue_t urls_queue = dispatch_queue_create("blog.devtang.com", null);
 dispatch_async(urls_queue, ^{ 
   // your code
 });
 dispatch_release(urls_queue);

另外,gcd还有一些高级用法,例如让后台两个线程并行执行,然后等两个线程都结束后,再汇总执行结果。这个可以用dispatch_group_t、dispatch_group_async、dispatch_group_notify来实现,示例如下:
复制代码 代码如下:

 // 合并汇总结果
 dispatch_group_t group = dispatch_group_create();
 dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{
      // 并行执行的线程一
 });
 dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{
      // 并行执行的线程二
 });
 dispatch_group_notify(group, dispatch_get_global_queue(0,0), ^{
      // 汇总结果
 });

dispatch队列不支持cancel(取消),没有实现dispatch_cancel()函数,不像nsoperationqueue,不得不说这是个小小的缺憾。