NSRunLoopMode
NSRunLoop
可以运行在不同的模式下,每种模式有不同的作用。
-
NSDefaultRunLoopMode
(Core Foundation中为kCFRunLoopDefaultMode
)
):默认设置,除了NSConnection
事件以外。 -
NSRunLoopCommonModes
:模式合集,默认包括Default、Modal、Event Tracking三大模式,可以处理几乎所有事件 -
UITrackingRunLoopMode
:在拖动或其它用户交互状态下处于该模式下,此时会限制输入事件的处理,例如拖动UIScrollView
会处于此种模式 -
NSConnectionReplyMode
:处理NSConnection相关事件,开发者一般用不到 -
NSModalPanelRunLoopMode
:用于macOS上,用于处理modal panels事件,NSSavePanel
或NSOpenPanel
-
NSEventTrackingRunLoopMode
:用于macOS上,用于处理拖拽和用户交互的模式
备注:
NSRunLoop
会根据当前线程的状态自动切换,例如
- 当主线程中有
UIScrollView
在滑动时,此时NSRunLoop
为UITrackingRunLoopMode
模式
- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
NSLog(@"ScrollView正在滑动");
NSLog(@"runloopMode:%@",[NSRunLoop currentRunLoop].currentMode);
//runloopMode:UITrackingRunLoopMode
}
- 当主线程中的
UIScrollView
未滑动时,此时NSRunLoop
为kCFRunLoopDefaultMode
(实验验证时,该状态的获取是通过点击button测试得到的)
- (void)greenBtnPressed:(UIButton *)btn {
NSLog(@"runloopMode:%@",[NSRunLoop currentRunLoop].currentMode);
}
实例1
NSTimer
默认运行在default模式下,如果此时一直拖动UIScrollView
,NSTimer
会暂停运行,直到松开UIScrollView
。
验证
NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:0.5 repeats:YES block:^(NSTimer * _Nonnull timer) {
static NSInteger i = 0;
i ++;
[btn setTitle:[NSString stringWithFormat:@"%ld", (long)i] forState:UIControlStateNormal];
}];
[timer fire];
NSTimer
每隔0.5秒更改button的title,如果此时滑动UIScrollView
,可以发现button的title未能变化,直到松开UIScrollView
解决方法
- 将NSTimer加入到
NSRunLoopCommonModes
中:[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
- 将NSTimer加入到另一个新的线程中,然后开启新线程的runloop。
实例2
NSURLConnection
也是默认运行在default模式下,不过我暂未实验检验