GCD的队列和任务理解
2017-05-18 本文已影响90人
泰好笑勒
任务和队列
GCD 的两个核心:任务和队列。
任务:要执行的操作或方法函数。
队列:存放任务的集合,而我们要做的就是将任务添加到队列然后执行,GCD会自动将队列中的任务按先进先出的方式取出并交给对应线程执行。
1. 任务
加入任务时有两种形式:同步任务和异步任务。
同步任务:不会开启新的线程。会阻塞当前线程。
异步任务:会开启新的线程。不会阻塞当前线程。
2. 队列
队列有两种队列,串行队列和并行队列。
串行队列:是指队列中的任务是一个接一个的执行,队首的任务执行完毕后才能执行其后面的任务,直至执行队尾的任务。尽可能使用一个线程。
并行队列:是指队列中的任务可以并发的执行,即开始执行队首的任务后,不必等其执行完毕就可以接着开始执行队首之后的任务,因此在某一个时刻可能存在同时执行的多个任务。
队列和任务的四种组合方式:(swift)
// attributes 参数被指定为 concurrent 时,该队列为并发队列,不设置时,默认串行队列
let concurrentQueue = DispatchQueue(label: "com.concurrent.queue", qos: .utility, attributes: .concurrent)
let serailQueue = DispatchQueue(label: "com.serail.queue")
1.串行同步
func serialWithSync() {
serailQueue.sync {
print(Thread.current)
for i in 0 ..< 3 {
print("aa = ",i)
}
}
serailQueue.sync {
print(Thread.current)
for i in 0 ..< 3 {
print("bb = ",i)
}
}
serailQueue.sync {
print(Thread.current)
for i in 0 ..< 3 {
print("cc = ",i)
}
}
print("end1")
}
// <NSThread: 0x600000071d80>{number = 1, name = main}
// aa = 0
// aa = 1
// aa = 2
// <NSThread: 0x600000071d80>{number = 1, name = main}
// bb = 0
// bb = 1
// bb = 2
// <NSThread: 0x600000071d80>{number = 1, name = main}
// cc = 0
// cc = 1
// cc = 2
// end1
分析:串行,顺序执行。同步,不会开启新的线程。
2.串行异步
func serialWithAsync() {
serailQueue.async {
print(Thread.current)
for i in 0 ..< 3 {
print("aa = ",i)
}
}
serailQueue.async {
print(Thread.current)
for i in 0 ..< 3 {
print("bb = ",i)
}
}
serailQueue.async {
print(Thread.current)
for i in 0 ..< 3 {
print("cc = ",i)
}
}
print("end2,",Thread.current)
}
// <NSThread: 0x608000261c00>{number = 3, name = (null)}
// aa = 0
// aa = 1
// end2, <NSThread: 0x608000074f40>{number = 1, name = main}
// aa = 2
// <NSThread: 0x608000261c00>{number = 3, name = (null)}
// bb = 0
// bb = 1
// bb = 2
// <NSThread: 0x608000261c00>{number = 3, name = (null)}
// cc = 0
// cc = 1
// cc = 2
分析:串行队列中的任务按顺序执行,然异步任务会开启新的线程,所以三个任务是在新的线程中依次执行。且异步任务不会阻塞当前线程(main),所以结束标志语句在主线程中不受阻塞执行。
3.并行同步
func concurrentWithSync() {
concurrentQueue.sync {
print(Thread.current)
for i in 0 ..< 3 {
print("aa = ",i)
}
}
concurrentQueue.sync {
print(Thread.current)
for i in 0 ..< 3 {
print("bb = ",i)
}
}
concurrentQueue.sync {
print(Thread.current)
for i in 0 ..< 3 {
print("cc = ",i)
}
}
print("end3")
}
// <NSThread: 0x608000071e80>{number = 1, name = main}
// aa = 0
// aa = 1
// aa = 2
// <NSThread: 0x608000071e80>{number = 1, name = main}
// bb = 0
// bb = 1
// bb = 2
// <NSThread: 0x608000071e80>{number = 1, name = main}
// cc = 0
// cc = 1
// cc = 2
// end3
分析:并行,任务会并发执行。同步,不会开启新的线程,且会阻塞当前线程。此时虽然用的是并行队列,但同步任务并没有开启新的线程,所以并行执行也就无从说起,就会在当前线程按顺序完成任务。
4.并行异步
func concurrentWithAsync() {
concurrentQueue.async {
print(Thread.current)
for i in 0 ..< 3 {
print("aa = ",i)
}
}
concurrentQueue.async {
print(Thread.current)
for i in 0 ..< 3 {
print("bb = ",i)
}
}
concurrentQueue.async {
print(Thread.current)
for i in 0 ..< 3 {
print("cc = ",i)
}
}
print("end4,",Thread.current)
}
// aa, <NSThread: 0x600000263380>{number = 3, name = (null)}
// end4, <NSThread: 0x600000077700>{number = 1, name = main}
// bb, <NSThread: 0x600000263800>{number = 4, name = (null)}
// aa = 0
// cc, <NSThread: 0x6000002636c0>{number = 5, name = (null)}
// bb = 0
// aa = 1
// cc = 0
// aa = 2
// cc = 1
// bb = 1
// cc = 2
// bb = 2
分析:异步任务负责开启新的线程,并发队列负责在新的线程中多任务执行。
类似的学习demo,建议使用playground,很是方便。且能顺便学习下swift,哈。