GCD详细使用笔记

第一、通过GCD创建队列(Queue)  

创建队列方式有以下几种:

  1、创建一个串行队列:

 dispatch_queue_t queue =dispatch_queue_create("串行队列", DISPATCH_QUEUE_SERIAL);

  2、创建一个并行队列:

 dispatch_queue_t queue =dispatch_queue_create("并行队列", DISPATCH_QUEUE_CONCURRENT);

  3、获取一个全局的队列:

 // 第1个参数表示队列执行的优先级:
// DISPATCH_QUEUE_PRIORITY_HIGH-高优先级
// DISPATCH_QUEUE_PRIORITY_DEFAULT-中优先级
// DISPATCH_QUEUE_PRIORITY_LOW-低优先级
// DISPATCH_QUEUE_PRIORITY_BACKGROUND-最低优先级
// 第2个参数暂不支持,传0即可;
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, );

  4、获取一个运行在主线程中的队列:

 dispatch_queue_t queue = dispatch_get_main_queue();

第二、队列的使用

1、执行一个串行队列:串行队列的特点是当前一个任务执行完毕后,后一个任务才开始执行。

 dispatch_queue_t queue =dispatch_queue_create("串行队列", DISPATCH_QUEUE_SERIAL);

     dispatch_async(queue, ^{

         NSLog(@"任务1 start");
sleep();
NSLog(@"任务1 end");
}); dispatch_async(queue, ^{ NSLog(@"任务2 start");
sleep();
NSLog(@"任务2 end");
});

2、执行一个并行队列:并行队列的特点是可以同时执行多个任务,在执行并行队列的时候,利用了信号量来控制同时执行的线程个数。

 dispatch_queue_t queue =dispatch_queue_create("并行队列", DISPATCH_QUEUE_CONCURRENT);
dispatch_semaphore_t semaphore =dispatch_semaphore_create();//这里指定可以同时执行的任务个数 dispatch_async(queue, ^{ dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"任务1 start");
sleep();
NSLog(@"任务1 end");
dispatch_semaphore_signal(semaphore);
}); dispatch_async(queue, ^{ dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"任务2 start");
sleep();
NSLog(@"任务2 end");
dispatch_semaphore_signal(semaphore);
}); dispatch_async(queue, ^{ dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"任务3 start");
sleep();
NSLog(@"任务3 end");
dispatch_semaphore_signal(semaphore);
});
dispatch_async(queue, ^{ dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"任务4 start");
sleep();
NSLog(@"任务4 end");
dispatch_semaphore_signal(semaphore);
});

3、执行一个并行队列:上面我们已经讲述了如何执行一个并行队列,并且指定并行任务最大个数,那么下面还有一个方法可以实现,就是利用

dispatch_barrier_async(并行队列中设置分界线),分界线的作用就是,将一组并行任务切割成多组并行任务顺序执行。我们也经常使用它来解决由于并发导致的数据不同步问题。

 dispatch_queue_t queue = dispatch_queue_create("并行队列", DISPATCH_QUEUE_CONCURRENT);

     dispatch_async(queue, ^{

         NSLog(@"任务1 start");
sleep();
NSLog(@"任务1 end");
}); dispatch_async(queue, ^{ NSLog(@"任务2 start");
sleep();
NSLog(@"任务2 end");
}); dispatch_barrier_async(queue, ^{ NSLog(@"--------------前面的并发任务执行完毕,准备执行下面的并发任务--------------");
sleep();
}); dispatch_async(queue, ^{ NSLog(@"任务3 start");
sleep();
NSLog(@"任务3 end");
}); dispatch_barrier_async(queue, ^{ NSLog(@"--------------前面的并发任务执行完毕,准备执行下面的并发任务--------------");
sleep();
}); dispatch_async(queue, ^{ NSLog(@"任务4 start");
sleep();
NSLog(@"任务4 end");
});
dispatch_async(queue, ^{ NSLog(@"任务5 start");
sleep();
NSLog(@"任务5 end");
});
dispatch_async(queue, ^{ NSLog(@"任务6 start");
sleep();
NSLog(@"任务6 end");
});

4、执行一个串行队列,并且我想知道这组队列什么时候执行完毕:

 dispatch_queue_t queue =dispatch_queue_create("串行队列", DISPATCH_QUEUE_SERIAL);
dispatch_group_t group =dispatch_group_create();
dispatch_group_async(group, queue, ^{
NSLog(@"任务1 start");
sleep();
NSLog(@"任务1 end");
});
dispatch_group_async(group, queue, ^{
NSLog(@"任务2 start");
sleep();
NSLog(@"任务2 end");
});
dispatch_group_async(group, queue, ^{
NSLog(@"任务3 start");
sleep();
NSLog(@"任务3 end");
}); // //方法一:
// dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
// //这里会等待group中的所有任务执行完成后才会向下执行
// NSLog(@"所有任务全部执行完毕"); //方法二:
dispatch_group_notify(group, queue, ^{ NSLog(@"所有任务全部执行完毕");
});
//这里不会等到group中所有的任务执行完成后才会向下执行
NSLog(@"所有任务还没执行完毕");

5、执行一个并行队列,并且我想知道这组队列什么时候执行完毕:

 dispatch_queue_t queue =dispatch_queue_create("并行队列", DISPATCH_QUEUE_CONCURRENT);
dispatch_semaphore_t semaphore =dispatch_semaphore_create();//这里指定可以同时执行的任务个数
dispatch_group_t group =dispatch_group_create();
dispatch_group_async(group, queue, ^{ dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"任务1 start");
sleep();
NSLog(@"任务1 end");
dispatch_semaphore_signal(semaphore);
});
dispatch_group_async(group, queue, ^{ dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"任务2 start");
sleep();
NSLog(@"任务2 end");
dispatch_semaphore_signal(semaphore);
});
dispatch_group_async(group, queue, ^{ dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"任务3 start");
sleep();
NSLog(@"任务3 end");
dispatch_semaphore_signal(semaphore);
});
dispatch_group_async(group, queue, ^{ dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
NSLog(@"任务4 start");
sleep();
NSLog(@"任务4 end");
dispatch_semaphore_signal(semaphore);
}); // //方法一:
// dispatch_group_wait(group, DISPATCH_TIME_FOREVER);
// //这里会等待group中的所有任务执行完成后才会向下执行
// NSLog(@"所有任务全部执行完毕"); //方法二:
dispatch_group_notify(group, queue, ^{ NSLog(@"所有任务全部执行完毕");
});
//这里不会等到group中所有的任务执行完成后才会向下执行
NSLog(@"所有任务还没执行完毕");

第三、利用GCD实现单例模式

 - (MySingletonObj *)shareObj{

     static MySingletonObj *singleton;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
//这里在整个程序的生命周期内只会执行一次
singleton =[[MySingletonObj alloc] init];
});
return singleton;
}

第四、利用GCD实现一个定时任务

 dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)( * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

      NSLog(@"3秒钟到了,我开始执行了");
});

第五、利用GCD实现一个循环操作:这个方法是同步运行的,类似于for循环

 dispatch_queue_t queue =dispatch_queue_create("串行队列", DISPATCH_QUEUE_SERIAL);
// dispatch_queue_t queue =dispatch_queue_create("并行队列", DISPATCH_QUEUE_CONCURRENT);
dispatch_apply(, queue, ^(size_t index) { //注意:如果queue是串行队列,那么index是按照0、1、2、3、4的顺序执行
// 如果queue是并行队列,那么index就不一定是按照顺序执行了
NSLog(@"%@", @(index));
});

第六、某些情况下,我们可能会想让Queue暂时停止一下,然后在某个时刻恢复处理,这时就可以使用dispatch_suspend以及dispatch_resume函数。暂停时,如果已经有block正在执行,那么不会对该block的执行产生影响。dispatch_suspend只会对还未开始执行的block产生影响。

 //暂停
dispatch_suspend(globalQueue)
//恢复
dispatch_resume(globalQueue)

以上是关于GCD中常用的一些函数和用法,其他方面有兴趣的可以到官网了解具体使用。

上一篇:c# winfrom 委托实现窗体相互传值


下一篇:初涉JavaScript模式 (3) : 字面量