多线程(一) 有关死锁以及串行并发同步异步概念
- 2019 年 10 月 8 日
- 筆記
多线程(一) 有关死锁以及串行并发同步异步概念
多线程
1、并发 串行 异步 同步
并发 串行 是指队列
异步同步 只是是否能开启线程
举个?
并发队列 (双行(多行)道路 可以超车 只要你可以)
可以让多个任务并发执行 (异步的时候可以超车 所以可以并发)
同步或者异步 (你有没有超车的能力) 是否具备开启新线程的能力
串行队列 (单行线)
任务一个接一个 无论同步还是异步 (无论有没有超车能力都只能一条线)
- ==队列(串行并发) 决定了运行方式==
- ==同步异步 决定了能否开启新线程的能力 (主队列只在主线程)==
2、死锁
案例1
题目:
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event { dispatch_async(dispatch_get_global_queue(0, 0), ^{ NSLog(@"1"); [self performSelector:@selector(test) withObject:nil afterDelay:0]; NSLog(@"3"); }); } - (void) test { NSLog(@"2"); }
执行结果:
2019-09-01 17:55:37.962948+0800 GCD[16903:233631] 1 2019-09-01 17:55:37.963250+0800 GCD[16903:233631] 3 复制代码
分析原因:
未执行原因:RunLoop未开启 消息添加到RunLoop中也不会被调用。
performSelector: withObject:
底层调用[self msgSend];
performSelector: withObject:afterDelay;
RunLoop相关
往RunLoop里面添加一个定时器
解决办法::
开启线程RunLoop
案例2
题目:
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event { NSThread *thread = [[NSThread alloc] initWithBlock:^{ NSLog(@"1"); }]; [thread start]; [self performSelector:@selector(test) onThread:thread withObject:nil waitUntilDone:YES]; NSLog(@"3"); } - (void) test { NSLog(@"2"); }
执行结果:
2019-09-01 18:23:51.670983+0800 GCD[17915:253178] 1 2019-09-01 18:23:51.674352+0800 GCD[17915:253102] *** Terminating app due to uncaught exception 'NSDestinationInvalidException', reason: '*** -[ViewController performSelector:onThread:withObject:waitUntilDone:modes:]: target thread exited while waiting for the perform' ...
分析原因:
block执行完毕后 线程没有开启RunLoop。所以线程销毁,销毁后再调用所以崩溃。
解决办法:
开启RunLoop
NSThread *thread = [[NSThread alloc] initWithBlock:^{ NSLog(@"1"); [[NSRunLoop currentRunLoop] addPort:[NSPort new] forMode:NSDefaultRunLoopMode]; [[NSRunLoop currentRunLoop] run]; }];