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中常用的一些函数和用法,其他方面有兴趣的可以到官网了解具体使用。
GCD详细使用笔记的更多相关文章
- Hbase技术详细学习笔记
注:转自 Hbase技术详细学习笔记 最近在逐步跟进Hbase的相关工作,由于之前对Hbase并不怎么了解,因此系统地学习了下Hbase,为了加深对Hbase的理解,对相关知识点做了笔记,并在组内进行 ...
- 基于【 MySql 】二 || mysql详细学习笔记
mysql重点学习笔记 /* Windows服务 */ -- 启动MySQL net start mysql -- 创建Windows服务 sc create mysql binPath= mysql ...
- MySQL 详细学习笔记 转
Windows服务 -- 启动MySQL net start mysql -- 创建Windows服务 sc create mysql binPath= mysqld_bin_path(注意:等号与值 ...
- 一千行 MySQL 详细学习笔记
Windows服务 -- 启动MySQL net start mysql -- 创建Windows服务 sc create mysql binPath= mysqld_bin_path(注意:等号与值 ...
- MySQL 详细学习笔记
Windows服务 -- 启动MySQL net start mysql -- 创建Windows服务 sc create mysql binPath= mysqld_bin_path(注意:等号与值 ...
- stl源码剖析 详细学习笔记 set map
// // set map.cpp // 笔记 // // Created by fam on 15/3/23. // // //---------------------------15/03 ...
- stl源码剖析 详细学习笔记 RB_tree (2)
//---------------------------15/03/22---------------------------- //一直好奇KeyOfValue是什么,查了下就是一个和仿函数差不多 ...
- stl源码剖析 详细学习笔记 RB_tree (1)
// // RB_tree_STL.cpp // 笔记 // // Created by fam on 15/3/21. // // #include "RB_tree_STL.h&q ...
- stl源码剖析 详细学习笔记priority_queue slist
// // priority_queue.cpp // 笔记 // // Created by fam on 15/3/16. // // //------------------------- ...
随机推荐
- 四轴飞行器1.3 MPU6050(大端)和M4的FPU开启方法
四轴飞行器1.3 MPU6050(大端)和M4的FPU开启方法 原创文章,欢迎转载,转载请注明出处 最近时间花在最多的地方就是STM32的I2C上了.之前就知道STM32的I2C并不好用, ...
- python基础学习笔记5--对象方法、属性和迭代器
对象方法.属性和迭代器 1.构造方法 1)构造方法和一般方法的不同点:当一个对象被创建后,会立即调用构造方法. 2)在Python中创建一个构造方法很容易,只要把init方法的名字从简单的init修改 ...
- RHEL4-Partition Image系统备份(软件版)
对于BBS,或Apache,PHP等相关网页的程序 备份: 1)/var/www/html目录,里面有PHP所写成的网页.此网页主要功能是从资料库中读取由信件存入的文章,或是使用者选择由网页输入资料时 ...
- 贴片陶瓷电容的NPO、C0G、X7R、X5R、Y5V、Z5U辨析
NPO与X7R.X5R.Y5V.Z5U神马的有啥区别?主要是介质材料不同.不同介质种类由于它的主要极化类型不一样,其对电场变化的响应速度和极化率亦不一样. 在相同的体积下的容量就不同,随之带来的电容器 ...
- 开源搜索技术—Lucene、Solr
Lucene是apache软件基金会4 jakarta项目组的一个子项目,是一个开放源代码的全文检索引擎工具包,即它不是一个完整的全文检索引擎,而是一个全文检索引擎的架构,提供了完整的查询引擎和索引引 ...
- BZOJ 2002 [Hnoi2010]Bounce 弹飞绵羊(动态树)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2002 [题目大意] 给出一片森林,操作允许更改一个节点的父亲,查询一个节点的深度. [ ...
- display:table标签来自动改变列宽 改变的同时table的整体宽度跟随变化
发现公司里的所有分页功能都是通过display:talbe来实现的,但是用户最近说要让表格列宽可以拖动:所有我就寻找了好多的办法:网上找了很多的资料,但是都不是我要的效果因为他们都是列宽不改变要不就是 ...
- python property装饰器
直接上代码: #!/usr/bin/python #encoding=utf-8 """ @property 可以将python定义的函数“当做”属性访问,从而提供更加友 ...
- Lowest Common Ancestor of a Binary Tree, with Parent Pointer
Given a binary tree, find the lowest common ancestor of two given nodes in tree. Each node contains ...
- [转]swift 学习资源 大集合
今天看到了一个swift的学习网站,里面收集了很多学习资源 [转自http://blog.csdn.net/sqc3375177/article/details/29206779] Swift 介绍 ...