Functions and queues
Adds the task to the queue and specifies the function that executes the task
- A task is encapsulated with a block, and the block of the task has no parameters and no return value
- A function that performs the task
- asynchronous
dispatch_async{}
- The next statement can be executed without waiting for the current statement to complete
- Opens the thread to execute the block’s task
- Asynchrony is a byword for multithreading
- synchronous
dispatch_sync{}
- You must wait for the current statement to complete before executing the next statement
- Thread will not be opened
- To restore the basic form of the current block task:
- asynchronous
Function dispatch_block_t block = ^{NSLog(@)"hello GCD"); }; Dispatch_queue_t queue = dispatch_queue_create("com.zb.cn", NULL); // dispatch_async(queue, block);Copy the code
Queue:
There are two special queues:
The home side columndispatch_get_main_queue()
- A queue used specifically to schedule tasks on the main thread is a serial queue
- Thread will not be opened
- If the main thread is currently running tasks, then whatever tasks are currently added to the main queue will not be scheduled
Global queuedispatch_get_global_queue()
- A global queue is a concurrent queue
- When using multithreaded development, if there is no special need for queues, you can use global queues directly when performing asynchronous tasks
Queues and functions:
After understanding the above combinations, try to work out the following task output sequence,,
- (void)textOne {dispatch_queue_t queue = dispatch_queue_create("zb", DISPATCH_QUEUE_CONCURRENT);
NSLog(@"Task 1");
dispatch_async(queue, ^{
NSLog(@Task 2 "");
dispatch_async(queue, ^{
NSLog(@"Task 3");
});
NSLog(@"The task 4");
});
NSLog(@"Task 5");
}
Copy the code
Task 1, dispatch_asyncblock task, task 5, so output task 1, task 5 in order; The nested asynchronous operation inside is exactly the same as the analysis outside, that is, the entire output order is 1, 5, 2, 4, 3
TextTwo {dispatch_queue_t queue = dispatch_queue_create("zb", DISPATCH_QUEUE_SERIAL);
NSLog(@"Task 1");
dispatch_async(queue, ^{
NSLog(@Task 2 "");
dispatch_sync(queue, ^{
NSLog(@"Task 3");
});
NSLog(@"The task 4");
});
NSLog(@"Task 5");
}
Copy the code
Here is a serial queue, dispatch_async tasks start a thread dedicated to processing, do not have to wait, so first output task 1, task 5 in sequence; Dispatch_asyncblock (dispatch_asyncblock (dispatch_asyncblock (dispatch_asyncblock (dispatch_asyncblock (dispatch_asyncblock (dispatch_asyncblock (dispatch_asyncblock))))) The block task is a synchronization function, so when task 2 is finished, the task 3 is added to the queue for execution, and the tasks in the queue are task 2, dispatch_asyncblock task, task 4, task 3. According to the FIFO principle, dispatch_asyncblock is executed after task 2 is completed. However, due to synchronization, task 3 must be executed before task 3 is executed, and task 4 must be executed before task 4 is executed. There is a deadlock here. So the output sequence of tasks is task 1, task 5, task 2, and then crash.
Deadlock generation
- The main thread is waiting to execute the task first because of the synchronization function
- The main queue is waiting for the main thread to finish executing its own task
- Main queue and main thread waiting on each other cause deadlock
What would the output look like if we changed the serial queue above to a concurrent queue?
Let’s look at an interview question
- (void)viewDidLoad {
[super viewDidLoad];
int a = 0;
while (a < 10) {
dispatch_async(dispatch_get_global_queue(0, 0), ^{
a ++;
});
}
NSLog(@"%d",a);
}
Copy the code
Problem 1: a++ error causes? Fault 2: After the modification is correct, the output a=? How can the output of A be 10 without changing the function and the queue?
__block is used to initialize a by copying Pointers and values from the stack to the struct. In a while loop, each time a thread is created, it performs an asynchronous operation and does not wait for the next task. This is also a time-consuming operation, so a++ operation can be done many times in the loop
The specific code is as follows
- (void)viewDidLoad { [super viewDidLoad]; __block int a = 0; // dispatch_semaphoRE_t lock = dispatch_semaphore_create(1);while(a < 10) { dispatch_async(dispatch_get_global_queue(0, 0), ^{ a ++; dispatch_semaphore_signal(lock); }); // dispatch_semaphore_wait(lock, DISPATCH_TIME_FOREVER); } NSLog(@"%d",a);
}
Copy the code
The use of the GCD
Barrier functiondispatch_barrier_sync
- (void)demo2{
dispatch_queue_t concurrentQueue = dispatch_queue_create("cooci", DISPATCH_QUEUE_CONCURRENT); */ dispatch_async(concurrentQueue, ^{for (NSUInteger i = 0; i < 5; i++) {
NSLog(@"download1-%zd-%@",i,[NSThread currentThread]); }}); dispatch_async(concurrentQueue, ^{for (NSUInteger i = 0; i < 5; i++) {
NSLog(@"download2-%zd-%@",i,[NSThread currentThread]); }}); /* 2. The dispatch_barrier_sync(concurrentQueue, ^{NSLog(@"-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- - % @ -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --",[NSThread currentThread]);
});
NSLog(@"Load so much, take a breath!!"); */ dispatch_async(concurrentQueue, ^{for (NSUInteger i = 0; i < 5; i++) {
NSLog(@"Routine treatment 3-% Zd - @,i,[NSThread currentThread]); }}); NSLog(@"********** dry up!!");
dispatch_async(concurrentQueue, ^{
for (NSUInteger i = 0; i < 5; i++) {
NSLog(@"Routine handling",i,[NSThread currentThread]); }}); }Copy the code
Download1 and download2 tasks are completed before daily processing 3 and 4 tasks are performed.
Question 1: If dispatch_queue_create(” cooCI “, DISPATCH_QUEUE_CONCURRENT); Dispatch_get_global_queue (0,0); What will the effect be?
After executing the code, you will find that the fence effect magically fails.
Dispatch_get_global_queue = dispatch_get_global_queue = dispatch_get_global_queue = dispatch_get_global_queue GG.
Question 2: what happens if the download1 or download2 task queue is replaced with another queue?
When you execute the code, you’ll also see that the fence doesn’t work if it’s not in the same queue, so that’s another thing to be careful about, you have to be in the same queue.
This is the first function of the fence function, to ensure sequential execution.
Look at the following code:
for (int i = 0; i < 5000; i++) {
dispatch_async(concurrentQueue, ^{
NSString *imageName = [NSString stringWithFormat:@"%d.jpg", (i % 10)];
NSURL *url = [[NSBundle mainBundle] URLForResource:imageName withExtension:nil];
NSData *data = [NSData dataWithContentsOfURL:url];
UIImage *image = [UIImage imageWithData:data];
[self.mArray addObject:image];
});
}
Copy the code
After execution, crash occurs, asynchronous function is created, multiple threads are created, and addObject operation is performed on array simultaneously, causing resource snatch, send crash.
dispatch_queue_t concurrentQueue = dispatch_queue_create("zb", DISPATCH_QUEUE_CONCURRENT);
for (int i = 0; i < 5000; i++) {
dispatch_async(concurrentQueue, ^{
.
.
.
dispatch_barrier_async(concurrentQueue, ^{
[self.mArray addObject:image];
});
});
}
Copy the code
After the command is executed, it can be executed normally and the output is displayed. The second function is to ensure thread safety.
Scheduling groupgroup
Create a group. Dispatch_group_create Group task. Dispatch_group_async Group task execution is complete. Dispatch_group_notify Wait time for dispatch_group_wait or dispatch_group_enter Dispatch_group_leave
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, queue, ^{
NSLog(@"Task 1");
});
long timeOut = dispatch_group_wait(group, 0.5);
dispatch_group_notify(group, queue, ^{
NSLog(@Task 2 ""); }); Or dispatch_group_enter (group); dispatch_async(queue, ^{ NSLog(@"Task");
dispatch_group_leave(group);
});
Copy the code
Enter and leave must be used in pairs; otherwise, crash may occur.
A semaphoredispatch_semaphore_t
Create the dispatch_SEMapHORE_create semaphore waits for dispatch_SEMapHORE_WAIT semaphore to release dispatch_semapHORE_signal
Can be used as a lock, as used at the beginning of this article, and can control the maximum number of concurrent GCD dispatch_semaphore_create(x)