preface
The Demo of this article can go to my Github MultiThreadDemo to view the source code, if there is any improper, I hope you point out.
Here is my last article about multithreading knowledge record iOS multithreading record (a).
GCD
introduce
GCD is a multithreaded programming solution developed by Apple. Through a simple API, we can create new threads to perform the tasks we need to perform. We do not need to manually create and manage threads, but only need to create queues and the corresponding functions to use.
advantages
- GCD can be used for parallel computing with multiple cores
- – GCDS automatically utilize more CPU cores (such as dual core, quad core)
- GCD automatically manages thread lifecycles (thread creation, task scheduling, thread destruction)
- The programmer only needs to tell the COMMUNIST party what task it wants to perform, without writing any thread management code
The core concept
- The queue
The queue here refers to the waiting queue for executing tasks, that is, the queue for storing tasks. A queue is a special linear table that uses FIFO (first in, first out), meaning that new tasks are always inserted at the end of the queue and read from the head of the queue. Each time a task is read, a task is released from the queue.
There are two types of queues in GCD: serial queues and concurrent queues. The main differences between the two are: the execution order is different, and the number of threads started is different.
-
Serial queues
- Only one task is executed at a time. Let the tasks be executed one after another. (Only one thread is started. After one task is completed, the next task is executed.)
-
Concurrent queue
- Multiple tasks can be executed concurrently. (Multiple threads can be opened and tasks can be executed simultaneously)
The concurrency capabilities of concurrent queues are only available for asynchronous functions.
- task
That’s what we need to do. There are two ways to execute tasks: sync and Async. The main differences between the two are whether to wait for queued tasks to finish and whether to have the ability to start a new thread.
- Sync Execution
- Synchronously add a task to a specified queue. The task will be executed after the previous task is completed.
- You can only execute tasks in the current thread and do not have the ability to start new threads.
- Asynchronous execution
- Asynchronously adds a task to a specified queue. It does not wait and executes the task directly.
- You can execute tasks in new threads and have the ability to start new threads.
The basic use
- Create a queue
- Adds a task to the queue
Create a queue
- Gets the main queue
// Get the main queuelet mainQueue = DispatchQueue.main
Copy the code
- Get global queue
// Get the global queuelet globalQueue = DispatchQueue.global()
Copy the code
- Create a queue
- Simple way to create
Specify the name of the queue, other defaults, such an initialized queue has a default configuration item, the default queue is a serial queue.
let queue = DispatchQueue(label: "com.jiangT.queue")
Copy the code
- Property setting mode creation
let queue = DispatchQueue.init(label: "com.jiangT.queue",
qos: DispatchQoS.default,
attributes:DispatchQueue.Attributes.concurrent,
autoreleaseFrequency:DispatchQueue.AutoreleaseFrequency.inherit,
target: nil)
Copy the code
Parameter Description:
- Label: Identifier of a queue. It is easy to distinguish queues for debugging.
- Qos: queue priority.
The priorities are unspecified, ranging from the lowest background to the highest userInteractive.
Background: lowest priority, equivalent to DISPATCH_QUEUE_PRIORITY_BACKGROUND. Hidden from the user, for example, large amounts of data are stored in the background
Utility: the priority of DISPATCH_QUEUE_PRIORITY_LOW is the same as that of DISPATCH_QUEUE_PRIORITY_LOW. For example: downloading a large file, networking, computing
Default: default priority. The priority is the same as DISPATCH_QUEUE_PRIORITY_DEFAULT. You are advised to use the default priority in most cases
UserInitiated: Priority is the same as DISPATCH_QUEUE_PRIORITY_HIGH and requires immediate results
UserInteractive: User interaction is relevant. Tasks need to be executed immediately for a good user experience. Use this priority for UI updates, event handling, and small workload tasks performed in the main thread.
Qos specifies the priority of queue work, and the system will schedule work according to the priority. The higher the priority is, the faster the execution will be, but the function will also be consumed. Therefore, accurately specifying the priority can ensure the effective use of resources by app.
- Attributes: Attributes of a queue that specify whether it is concurrent or serial.
- AutoreleaseFrequency: indicates the frequency of automatic release. Some queues are automatically released after the task is completed. Others are not automatically released and need to be released manually.
Add tasks
- Synchronization task
let globalQueue = DispatchQueue.global()
globalQueue.sync {
print("sync + \(Thread.current)")}Copy the code
- Asynchronous tasks
let globalQueue = DispatchQueue.global()
globalQueue.async {
print("async + \(Thread.current)")}Copy the code
It only takes two steps to use GCD, but since we have two types of queue (serial queue/concurrent queue) and two types of task execution (synchronous/asynchronous execution), we have four different combinations. Plus two special queues: global concurrent queues and primary queues. Global concurrent queues can be used as normal concurrent queues. But the main queue is a little bit special, so we have two more combinations. So there are six different combinations.
- Concurrent queue + synchronous execution
- Concurrent queue + asynchronous execution
- Serial queue + synchronous execution
- Serial queue + asynchronous execution
- Main queue + synchronous execution
- Main queue + asynchronous execution
The result of various combinations
The difference between | Concurrent queue | Serial queues | The home side column |
---|---|---|---|
Synchronization (sync) | No new thread is started and the task is executed sequentially | No new thread is started and the task is executed sequentially | Can cause a deadlock |
Asynchronous (async) | Start a new thread and execute tasks concurrently | Start a new thread (1 thread) to execute tasks in serial | No new thread is started and the task is executed sequentially |
Concurrent queue + synchronous execution
DispatchQueue.global().sync {
print("sync1 + \(Thread.current)")
}
DispatchQueue.global().sync {
print("sync2 + \(Thread.current)")} ----- ----- sync1 + <NSThread: 0x600003bc35c0>{number = 1, name = main} sync2 + <NSThread: 0x600003bc35c0>{number = 1, name = main}Copy the code
Conclusion:
- All tasks are executed in the current thread (main thread) and no new threads are started (synchronous execution does not have the ability to start new threads).
- Tasks are performed sequentially. Why sequential: Although concurrent queues can open multiple threads and perform multiple tasks at the same time. But synchronization tasks do not have the ability to start new threads, so there is no concurrency. The current thread must wait for the tasks in the queue to complete before it can continue to perform the following operations (synchronization tasks need to wait for the tasks in the queue to complete). So tasks can only be executed sequentially, not simultaneously.
Concurrent queue + asynchronous execution
DispatchQueue.global().async {
print("begin")
print("async1 + \(Thread.current)")
}
DispatchQueue.global().async {
print("begin")
print("async2 + \(Thread.current)")
}
DispatchQueue.global().async {
print("begin")
print("async3 + \(Thread.current)")
}
DispatchQueue.global().async {
print("begin")
print("async4 + \(Thread.current)")
}
DispatchQueue.global().async {
print("begin")
print("async5 + \(Thread.current)")} ----- Command output: ----- begin BEGIN begin async1 + <NSThread: 0x60000018e540>{number = 3, name = (null)} begin async2 + <NSThread: 0x600000196600>{number = 4, name = (null)} begin async3 + <NSThread: 0x6000001bc900>{number = 5, name = (null)} async5 + <NSThread: 0x600000196600>{number = 4, name = (null)} async4 + <NSThread: 0x60000018e540>{number = 3, name = (null)}Copy the code
Conclusion:
- In addition to the current thread (the main thread), three more threads are opened and tasks are executed alternately/simultaneously. Asynchronous execution has the ability to start new threads. And concurrent queues can open multiple threads to perform multiple tasks at the same time).
Serial queue + synchronous execution
let queue = DispatchQueue.init(label: "")
queue.sync {
print("sync1 + \(Thread.current)")
}
queue.sync {
print("sync1 + \(Thread.current)"----- ----- sync1 + <NSThread: 0x600000a968c0>{number = 1, name = main} sync2 + <NSThread: 0x600000a968c0>{number = 1, name = main}Copy the code
Conclusion:
- All tasks are executed in the current thread (main thread) and no new thread is started (synchronous execution does not have the ability to start a new thread).
- Tasks are executed sequentially (only one task is executed in a serial queue at a time, and tasks are executed sequentially one after another).
Serial queue + asynchronous execution
let queue = DispatchQueue.init(label: "")
queue.async {
print("begin")
print("async1 + \(Thread.current)")
}
queue.async {
print("begin")
print("async2 + \(Thread.current)")
}
queue.async {
print("begin")
print("async3 + \(Thread.current)")} ----- The command output is as follows: ----- begin async1 + <NSThread: 0x60000032B940 >{number = 3, name = (null)} begin async2 + <NSThread: 0x60000032b940>{number = 3, name = (null)} begin async3 + <NSThread: 0x60000032b940>{number = 3, name = (null)}Copy the code
Conclusion:
- A new thread is started (asynchronous execution has the ability to start new threads, serial queues only start one thread).
- Tasks are executed sequentially (only one task is executed in a serial queue at a time, and tasks are executed sequentially one after another).
Main queue + synchronous execution
- Main queue: A special serial queue that comes with the GCD
// The main thread performs the synchronization task funcsyncMain() {
print("Method start") // After calling the method, you can see the output dispatchqueue.main.sync {print("Will cause a deadlock.")}} ----- Output: ----- Method startCopy the code
Conclusion:
- When a task is put into the main queue, it waits for the main queue to finish executing the current task.
- The main thread is now processing the syncMain method, and the task needs to wait for syncMain to finish executing.
- When syncMain is executed, it waits for the task to complete before completing the method.
- The syncMain method and task then start waiting for each other, creating a deadlock.
Main queue + asynchronous execution
DispatchQueue.main.async {
print("begin")
print("async1 + \(Thread.current)")
}
DispatchQueue.main.async {
print("begin")
print("async2 + \(Thread.current)")
}
DispatchQueue.main.async {
print("begin")
print("async3 + \(Thread.current)")
}
DispatchQueue.main.async {
print("begin")
print("async4 + \(Thread.current)")
}
DispatchQueue.main.async {
print("begin")
print("async5 + \(Thread.current)")} ----- Command output: ----- begin async1 + <NSThread: 0x600001AF6880 >{number = 1, name = main} begin async2 + <NSThread: 0x600001af6880>{number = 1, name = main} begin async3 + <NSThread: 0x600001af6880>{number = 1, name = main} begin async4 + <NSThread: 0x600001af6880>{number = 1, name = main} begin async5 + <NSThread: 0x600001af6880>{number = 1, name = main}Copy the code
Conclusion:
- All tasks are executed in the current thread (main thread) and no new thread is started (asynchronous execution has the ability to start threads, but because it is the main queue, all tasks are in the main thread).
- Tasks are executed sequentially (because the main queue is a serial queue, only one task is executed at a time, and tasks are executed sequentially one after another).
Interthread communication
/** * dispatchqueue.global ().async {print("async-begin")
print("global + \(Thread.current)")
print("async-end")
let testNum = 666 dispatchqueue.main.async {print("main + \(Thread.current)")
print("pass testNum: \(testNum)")}} - output: -- -- -- -- -- async - begin global + < NSThread: 0x60000383d100>{number = 3, name = (null)} async-end main + <NSThread: 0x600003821680>{number = 1, name = main} passtestNum: 666
Copy the code
Conclusion:
- You can see that the global thread executes the task first, then returns to the main thread to perform the corresponding operation on the main thread and receives the global thread’s testNum.
DispatchWorkItem
DispatchWorkItem is used to help DispatchQueue perform tasks in a queue.
Typically, when we start an asynchronous thread, we create a queue like this and execute async methods to submit tasks as closures.
DispatchQueue.global().async {
// do async task
}
Copy the code
The DispatchWorkItem class is used to encapsulate the task as an object that performs the task.
let item = DispatchWorkItem {
// do task
}
DispatchQueue.global().async(execute: item)
Copy the code
Tasks can also be performed using the Perform method of the DispatchWorkItem instance object
let workItem = DispatchWorkItem {
// do task
}
DispatchQueue.global().async {
workItem.perform()
}
Copy the code
The fence method
We sometimes need to perform two sets of operations asynchronously, and after the first set of operations is complete, the second set of operations can be performed. We need a fence to separate two asynchronously executed action groups, which can contain one or more tasks.
The fence waits for all the tasks appended to the concurrent queue to complete before appending the specified task to the asynchronous queue.
let dataQueue = DispatchQueue(label: "com.jiangT.queue", attributes: .concurrent)
let item = DispatchWorkItem(qos: .default, flags: .barrier) {
print("barrier + \(Thread.current)")
dataQueue.async {
print("async1 + \(Thread.current)")
}
dataQueue.async {
print("async2 + \(Thread.current)")
}
}
dataQueue.async(execute: item)
dataQueue.async {
print("async3 + \(Thread.current)")
}
dataQueue.async {
print("async4 + \(Thread.current)"----- barrier + <NSThread: 0x600000F9D800 >{number = 3, name = (null)} async3 + <NSThread: 0x600000f9da40>{number = 6, name = (null)} async4 + <NSThread: 0x600000f9bac0>{number = 7, name = (null)} async1 + <NSThread: 0x600000f9d800>{number = 3, name = (null)} async2 + <NSThread: 0x600000fa4040>{number = 8, name = (null)}Copy the code
Nofify is notified of the completion of a task
let workItem = DispatchWorkItem {
// do async task
print(Thread.current)
}
DispatchQueue.global().async {
workItem.perform()
}
workItem.notify(queue: DispatchQueue.main) {
// update UI
print(thread. current)} ----- ----- <NSThread: 0x60000193CE40 >{number = 3, name = (null)} <NSThread: 0x600001921100>{number = 1, name = main}Copy the code
Use wait to wait for a task to complete
let queue = DispatchQueue(label: "queue", attributes: .concurrent)
let workItem = DispatchWorkItem {
sleep(5)
print("done")
}
queue.async(execute: workItem)
print("before waiting")
workItem.wait()
print("after waiting"----- ----- before waitingdone
after waiting
Copy the code
Delayed execution method
letdelay = DispatchTime.now() + DispatchTimeInterval.seconds(10) DispatchQueue.main.asyncAfter(deadline: Delay) {// delay} can be simplified as:letDelay = DispatchTime. Now () + 10 DispatchQueue. Main. AsyncAfter (deadline: delay) {/ / delay the}Copy the code
Other delayed execution methods:
func asyncAfter(deadline: DispatchTime, execute: DispatchWorkItem)
func asyncAfter(deadline: DispatchTime, qos: DispatchQoS, flags: DispatchWorkItemFlags, execute: () -> Void)
func asyncAfter(wallDeadline: DispatchWallTime, execute: DispatchWorkItem)
func asyncAfter(wallDeadline: DispatchWallTime, qos: DispatchQoS, flags: DispatchWorkItemFlags, execute: () -> Void)
Copy the code
Fast iterative method
The GCD dispatch_apply() call is now concurrentPerform(), and the GCD dispatch_apply() call is now executed concurrently 5 times
DispatchQueue.concurrentPerform(iterations: 5) {
print("\ [$0)")} ----- The command output is ----- 2 0 1 3 4Copy the code
Queue group operation
Sometimes there is a need to execute two time-consuming tasks asynchronously and then return to the main thread when both tasks are completed. At this point we can use the GCD queue group.
let queue = DispatchQueue.global()
let group = DispatchGroup()
group.enter()
queue.async(group: group) {
DispatchQueue.main.asyncAfter(deadline: .now() + 2, execute: {
print("Task one finished")
group.leave()
})
}
group.enter()
queue.async(group: group) {
print("Task two finished")
group.leave()
}
group.enter()
queue.async(group: group) {
print("Task three finished")
group.leave()
}
group.notify(queue: queue) {
print("All task has finished")} ---- Command output: ----- Task two Finished Task three finished Task one finished All Task has finishedCopy the code