gossip

This article mainly summarizes the actual work of multithreading, as well as their understanding of multithreading

Multithreaded usage

  • Thread
  • NSOperation
  • GCD
  • ,,,,

This article mainly summarizes some use methods of GCD.

The GCD introduction

- Grand Central Dispatch(GCD) is one of the technologies for performing tasks asynchronously. The thread management described in the application is implemented in the system using code. Because thread management is implemented as part of the system, it can be managed uniformly and tasks can be executed, which makes it more efficient. - GCD consists of tasks and queuesCopy the code

Concepts in the COMMUNIST Party of China

concept API understand
The queue Dispatch Queue A thread queue, similar to a task pool, can execute a defined task according to certain rules by adding it to the queue.
Asynchronous thread dispatch_async
Thread synchronization dispatch_sync Appending the task to the specified queue. The dispatch_sync function waits until the appended Block ends.
Barrier function diapatch_barrier_async The dispatch_sync function does not wait for tasks to be appended to the specified queue.
One-time function dispatch_once The added Block task is executed only once in the entire program life cycle
Thread group Dispatch Group
A semaphore Dispatch Semaphone
### DCG usage method
  • 1. Create a queue (serial queue, concurrent queue)
  • 2. Create a task and add the task to a queue. The system executes the task (synchronous or asynchronous) based on the task type.

The queue to create

  • Serial Dispatch Queue
    • 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 Dispatch Queue
    • Multiple tasks can be executed concurrently. Multiple threads can be started to perform multiple tasks at the same time
// dispatch_queue_t queue = dispatch_queue_create("com.queue.serial", DISPATCH_QUEUE_SERIAL); // dispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT);Copy the code

Task creation

  • asynchronous
Dispatch_async (queue, ^{// async tasks});Copy the code
  • synchronous
Dispatch_sync (queue, ^{// sync task});Copy the code

Because there are two ways to execute tasks and two ways to queue, there are multiple combinations;

The way tasks and queues are combined

Concurrent queue Serial queues The home side column
Synchronization (sync) Do not start a new thread, the task is executed sequentially Do not start a new thread, the task is executed sequentially In a deadlock, the task waits for the main thread to execute, but in synchronization, the main thread waits for the task to execute, causing a deadlock
Asynchronous (async) Start a new thread and execute tasks concurrently Start a new thread and execute tasks sequentially Do not start a new thread, the task is executed sequentially

GCD concrete implementation

  • Serial queue + asynchronous task
dispatch_queue_t queue = dispatch_queue_create("com.queue.serial", DISPATCH_QUEUE_SERIAL); Dispatch_async (queue, ^{NSLog(@" task 1--%@",[NSThread currentThread]); }); Dispatch_async (queue, ^{NSLog(@" task 2--%@",[NSThread currentThread]); }); Dispatch_async (queue, ^{NSLog(@" task 3--%@",[NSThread currentThread]); }); NSLog(@" task 4--%@",[NSThread currentThread]);Copy the code

Output result:

Objc [1491:2645294] objC [1491:2645294] task 1--<NSThread: 0x281490dc0>{number = 1, name = main} 0x2814C5680 >{number = 5, name = (null)} objc[1491:2645294] 0x2814C5680 >{number = 5, name = (null)} objC [1491:2645294] Task 3--<NSThread: 0x2814C5680 >{number = 5, name = (null)} objC [1491:2645294] Task 3--<NSThread: 0x2814C5680 >{number = 5, name = (null)}Copy the code
  • Serial queue + synchronization task
dispatch_queue_t queue = dispatch_queue_create("com.queue.serial", DISPATCH_QUEUE_SERIAL); Dispatch_sync (queue, ^{NSLog(@" task 1--%@",[NSThread currentThread]); }); Dispatch_sync (queue, ^{NSLog(@" task 2--%@",[NSThread currentThread]); }); Dispatch_sync (queue, ^{NSLog(@" task 3--%@",[NSThread currentThread]); }); NSLog(@" task 4--%@",[NSThread currentThread]);Copy the code

Output result:

Objc [1489:2644861] Task 1--<NSThread: 0x280C28dc0 >{number = 1, name = main} Objc [1489:2644861] task 2--<NSThread: 0x280C28dc0 > 0x280C28dc0 >{number = 1, name = main} objc[1489:2644861] 0x280C28dc0 >{number = 1, name = main} objc[1489:2644861] Task 4--<NSThread: 0x280C28dc0 >{number = 1, name = main}Copy the code
  • Concurrent queue + asynchronous execution
ispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT); Dispatch_async (queue, ^{NSLog(@" task 1--%@",[NSThread currentThread]); }); Dispatch_async (queue, ^{NSLog(@" task 2--%@",[NSThread currentThread]); }); Dispatch_async (queue, ^{NSLog(@" task 3--%@",[NSThread currentThread]); }); NSLog(@" task 4--%@",[NSThread currentThread]);Copy the code

Output result:

Objc [1497:2646937] ObjC [1497:2646937] task 2--<NSThread: 0x283E80dc0 >{number = 1, name = main} 0x283EDe540 >{number = 5, name = (null)} objc[1497:2646950] 0x283EDe540 >{number = 5, name = (null)} objC [1497:2646949] Task 1--<NSThread: 0x283ed1e80>{number = 4, name = (null)}Copy the code
  • Concurrent queue + synchronization task
dispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT); Dispatch_sync (queue, ^{NSLog(@" task 1--%@",[NSThread currentThread]); }); Dispatch_sync (queue, ^{NSLog(@" task 2--%@",[NSThread currentThread]); }); Dispatch_sync (queue, ^{NSLog(@" task 3--%@",[NSThread currentThread]); }); NSLog(@" task 4--%@",[NSThread currentThread]);Copy the code

The output

Objc [1501:2647457] ObjC [1501:2647457] task 1--<NSThread: 0x2811D4dc0 >{number = 1, name = main} task 2--<NSThread: 0x2811D4dc0 > 0x2811D4dc0 >{number = 1, name = main} objc[1501:2647457] 0x2811D4dc0 >{number = 1, name = main} objc[1501:2647457] Task 4--<NSThread: 0x2811D4dc0 >{number = 1, name = main}Copy the code
  • Main queue + asynchronous task
dispatch_queue_t queue = dispatch_get_main_queue(); Dispatch_async (queue, ^{NSLog(@" task 1--%@",[NSThread currentThread]); }); Dispatch_async (queue, ^{NSLog(@" task 2--%@",[NSThread currentThread]); }); Dispatch_async (queue, ^{NSLog(@" task 3--%@",[NSThread currentThread]); }); NSLog(@" task 4--%@",[NSThread currentThread]);Copy the code

Output result:

Objc [1504:2648185] task 1--<NSThread: 0x283544dc0>{number = 1, name = main} objc[1504:2648185] task 1--<NSThread: 0x283544dc0> 0x283544dc0>{number = 1, name = main} objc[1504:2648185] 0x283544dc0>{number = 1, name = main} objc[1504:2648185] Task 3--<NSThread: 0x283544dc0>{number = 1, name = main}Copy the code
  • Main queue + synchronization task
    • Program crashes, program cannot execute due to deadlock
dispatch_queue_t queue = dispatch_get_main_queue(); Dispatch_sync (queue, ^{NSLog(@" task 1--%@",[NSThread currentThread]); }); Dispatch_sync (queue, ^{NSLog(@" task 2--%@",[NSThread currentThread]); }); Dispatch_sync (queue, ^{NSLog(@" task 3--%@",[NSThread currentThread]); }); NSLog(@" task 4--%@",[NSThread currentThread]);Copy the code

Barrier function

– Set the fence blocker so that the tasks behind the fence will wait until the tasks in front of the fence finish executing

dispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT); Dispatch_async (queue, ^{NSLog(@" task 1--%@",[NSThread currentThread]); }); Dispatch_async (queue, ^{NSLog(@" task 2--%@",[NSThread currentThread]); }); dispatch_barrier_async(queue, ^{ for (int i = 0; i<4; I++) {NSLog (@ "task 5 - % @", [NSThread currentThread]); }}); dispatch_async(queue, ^{ for (int i = 0; i<4; I++) {NSLog (@ "task 3 - % @", [NSThread currentThread]); }}); NSLog(@" task 4--%@",[NSThread currentThread]);Copy the code

Output result:

Objc [1517:2651733] objC [1517:2651733] task 1--<NSThread: 0x283A74dc0 >{number = 1, name = main} 0x283A22880 >{number = 4, name = (null)} objc[1517:2651733] 0x283A22880 >{number = 4, name = (null)} objc[1517:2651733] 0x283A22880 >{number = 4, name = (null)} objc[1517:2651733] 0x283A22880 >{number = 4, name = (null)} objc[1517:2651733] 0x283A22880 >{number = 4, name = (null)} objc[1517:2651733] 0x283A22880 >{number = 4, name = (null)} objc[1517:2651733] 0x283A22880 >{number = 4, name = (null)} objc[1517:2651733] 0x283A22880 >{number = 4, name = (null)} objc[1517:2651733] 0x283A22880 >{number = 4, name = (null)} objC [1517:2651733] Task 3--<NSThread: 0x283A22880 >{number = 4, name = (null)}Copy the code

Linear function

static dispatch_once_t onceToken; Dispatch_once (&onceToken, ^{// task code});Copy the code

Thread group

  • The main application scenario of thread groups in practice is that specific operations need to be performed after the completion of multiple asynchronous operations
  • The dispatch_group_notify function notifies subsequent operations only after all dispatch_group_async tasks are completed

Scenario example: After loading multiple images in the background of the APP, the images are synthesized to form a new image

dispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT); NSLog(@" task 0--%@",[NSThread currentThread]); dispatch_group_t group = dispatch_group_create(); dispatch_group_async(group, queue, ^{ [NSThread sleepForTimeInterval:1]; NSLog(@" task 1--%@",[NSThread currentThread]); }); dispatch_group_async(group, queue, ^{ [NSThread sleepForTimeInterval:1]; NSLog(@" task 2--%@",[NSThread currentThread]); }); dispatch_group_async(group, queue, ^{ [NSThread sleepForTimeInterval:1]; NSLog(@" task 3--%@",[NSThread currentThread]); }); Dispatch_group_notify (dispatch_get_main_queue(), ^{NSThread sleepForTimeInterval:1]; NSLog(@" task 4--%@",[NSThread currentThread]); }); NSLog(@" task 5--%@",[NSThread currentThread]);Copy the code

The output

Objc [1527:2656709] objC 0--<NSThread: 0x2814e0dc0>{number = 1, name = main} 2020-11-20 11:26:452.652206 +0800 OBJC [1527:2656709] 0x2814e0dc0>{number = 1, name = main} 2020-11-20 11:26:46.657634+0800 OBJC [1527:2656718] 0x2814E7AC0 >{number = 3, name = (null)} 2020-11-20 11:26:46.657748+0800 OBJC [1527:2656719] 0x2814DB800 >{number = 5, name = (null)} 2020-11-20 11:26:46.657767+0800 OBJC [1527:2656720] 0x2814A8040 >{number = 6, name = (null)} 2020-11-20 11:26:47.659350+0800 OBJC [1527:2656709] 0x2814e0dc0>{number = 1, name = main}Copy the code
  • Thread groups can also be dispatch_group_enter or dispatch_group_leave
dispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT); NSLog(@" task 0--%@",[NSThread currentThread]); dispatch_group_t group = dispatch_group_create(); dispatch_group_enter(group); dispatch_async(queue, ^{ [NSThread sleepForTimeInterval:1]; NSLog(@" task 1--%@",[NSThread currentThread]); dispatch_group_leave(group); }); dispatch_group_enter(group); dispatch_async(queue, ^{ [NSThread sleepForTimeInterval:1]; NSLog(@" task 2--%@",[NSThread currentThread]); dispatch_group_leave(group); }); dispatch_group_enter(group); dispatch_async(queue, ^{ [NSThread sleepForTimeInterval:1]; NSLog(@" task 3--%@",[NSThread currentThread]); dispatch_group_leave(group); }); Dispatch_group_notify (dispatch_get_main_queue(), ^{NSThread sleepForTimeInterval:1]; NSLog(@" task 4--%@",[NSThread currentThread]); }); NSLog(@" task 5--%@",[NSThread currentThread]);Copy the code

The output

2020-11-20 11:34:55.314133+0800 Objc [1530:2658637] 0--<NSThread: 0x280F78DC0 >{number = 1, name = main} 2020-11-20 11:34:55.314234+0800 OBJC [1530:2658637] 0x280F78DC0 >{number = 1, name = main} 2020-11-20 11:34:56.319524+0800 OBJC [1530:2658651] 0x280F29340 >{number = 4, name = (null)} 2020-11-20 11:34:56.319513+0800 OBJC [1530:2658649] 0x280F29540 >{number = 3, name = (null)} 2020-11-20 11:34:56.319536+0800 OBJC [1530:2658650] 0x280F34B80 >{number = 6, name = (null)} 2020-11-20 11:34:57.321272+0800 OBJC [1530:2658637] 0x280f78dc0>{number = 1, name = main}Copy the code

A semaphore

  • This function is understood as a complement to fence functions and serial queues for more fine-grained control
// dispatch_semaphoRE_t semaphonre = dispatch_semaphore_create(1); dispatch_queue_t queue = dispatch_queue_create("com.queue.concurrent", DISPATCH_QUEUE_CONCURRENT); NSLog(@" operation started "); for (int i = 1; i < 10; i ++) { dispatch_async(queue, ^{ /* 1. 2. If the semaphore is >0, */ DISPATCH_TIME_FOREVER */ semaphore_wait(semaphonre, DISPATCH_TIME_FOREVER); [NSThread sleepForTimeInterval:2]; NSLog (@ "operation = = = % d", I); */ SemaphoRE_signal (semaphonre) */ Semaphore_signal (semaphonre) }); } NSLog(@" operation complete ");Copy the code

reference

  • Advanced Programming in Objective-C
  • Juejin. Cn/post / 684490…