一.理论
Runloop:字面意思:跑圈,实际他上是一个对象,这个对象管理了其需要处理的事件和消息,并提供了一个入口函数来执行do{}while{}逻辑,线程执行这个函数后,就会一直处于接受消息->等待->处理的循环中,直到这个循环结束,函数返回.
RunLoop就是线程中的一个循环,RunLoop在循环中会不断检测,通过Input sources(输入源)和Timer sources(定时源)两种来源等待接受事件;然后对接受到的事件通知线程进行处理,并在没有事件的时候进行休息。
OSX/iOS 系统中,提供了两个这样的对象:NSRunLoop (继承于后者非线程安全的)和 CFRunLoopRef(在CoreFoundation框架,所以是线程安全的)。
ios中开启线程很消耗性能,主线程要消耗1m,后台线程要消耗512k内存
CFRunLoopRef 的代码是开源的,你可以在这里 http://opensource.apple.com/tarballs/CF/CF-855.17.tar.gz
二.api
CFRunLoopRef runloop本类
CFRunLoopModeRef runloop运行模式 有5种模式
1. kCFRunLoopDefaultMode:默认mode,通常住线程在这个mode下运行
2.UITrackingRunLoopMode:追踪mode,保证Scrollview滑动流畅不受其他
影响
3.UIInitializationRunLoopMode:启动程序后的过渡mode,启动完成后就不
使用
4.GSEventReceivRunLoopMode: 一般用不到
5.kCFRunLoopCommonModes:占位mode,作为1和2标志用
CFRunLoopSourceRef runloop里面的内容—事件源,输入源
分为Source0
Source1两种
CFRunLoopTimerRef。runloop里面的内容—定时器
CFRunLoopObserverRef 观察者
一个 RunLoop 包含若干个 Mode,每个 Mode 又包含若干个 Source/Timer/Observer。每次调用 RunLoop 的主函数时,只能指定其中一个 Mode,这个Mode被称作 CurrentMode。如果需要切换 Mode,只能退出 Loop,再重新指定一个 Mode 进入。这样做主要是为了分隔开不同组的 Source/Timer/Observer,让其互不影响。
三.使用
NSTimer的使用解决定时器会延时)
一般项目中的首页头部会有(scrollerview+timer)+tableview的组合当我们滑动tableview时会出现滚动图停止滚动
这是因为我们把timer加入到主线程中的main runloop当中去了而主线程的是NSDefaultRunLoopMode当主线程中的runloop没有其他事件时那么timer正常运行,当main runloop接受到其他itme(scoure/其他事件)会切换成UITrackingRunLoopMode当滑动停止时将会切换成默认解决这个问题有
2种方式
1开启一个线程将定时器添加到子线程中的runloop当中
2.timer:有2种方式初始化,
//创建线程
NSThread *subThead = [[NSThread alloc]initWithTarget:self selector:@selector(timerTest) object:nil];
self.subThread = subThead;
//启动线程
[subThead start];
-(void)timerTest{
@autoreleasepool {
[NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timerUpdate) userInfo:nil repeats:YES];
[[NSRunLoop currentRunLoop] run];
}
}
刷新ui
- (void)timerUpdate
{
// NSLog(@”当前线程:%@”,[NSThread currentThread]);
// NSLog(@”启动RunLoop后–%@”,[NSRunLoop currentRunLoop].currentMode);
// NSLog(@”currentRunLoop:%@”,[NSRunLoop currentRunLoop]);
//由于在子线程中所以我们要在主线城中刷新ui
dispatch_async(dispatch_get_main_queue(), ^{
self.count –;
NSString *timerText = [NSString stringWithFormat:@”计时器:%ld”,self.count];
self.TimeLable.text = timerText;
});
}
timer有2种方式创建
第一种:scheduledTimerWithTimeInterval
第二种:imerWithTimeInterva我们利用第二种方式初始化timer然后加入到当前线程中使用
NSRunLoopCommonModes(占位)模式
self.timer = [NSTimer timerWithTimeInterval:0.01 target:self selector:@selector(addTime) userInfo:nil repeats:YES];
[[NSRunLoop currentRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
解决tableview滑动卡顿滑动以后加载(image推迟显示)
核心代码
//主要代码
[imageView performSelectorOnMainThread:@selector(setImage:) withObject:image waitUntilDone:NO modes:@[NSDefaultRunLoopMode]];
后台常驻线程(很常用)
self.subTherad = [[NSThread alloc]initWithTarget:self selector:@selector(run) object:nil];
[self.subTherad start];
(void)run{
//只要往RunLoop中添加了 timer、source或者observer就会继续执行,一个Run Loop通常必须包含一个输入源或者定时器来监听事件,如果一个都没有,Run Loop启动后立即退出。
//1、添加一个input source
[[NSRunLoop currentRunLoop] addPort:[NSPort port] forMode:NSDefaultRunLoopMode];
[[NSRunLoop currentRunLoop] run];
//2、添加一个定时器
// NSTimer *timer = [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(test) userInfo:nil repeats:YES];
// [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
// [[NSRunLoop currentRunLoop] run];
}
调用
[self performSelector:@selector(test) onThread:self.subTherad withObject:nil waitUntilDone:NO];
让奔溃的程序起死回生
参考:https://www.jianshu.com/p/fee0c5155b8e
四.与线程的关系
线程和 RunLoop 之间是一一对应的,其关系是保存在一个全局的 Dictionary 里(key是thread,value是runloop,安卓上面也有looper)。线程刚创建时并没有 RunLoop,如果你不主动获取,那它一直都不会有。RunLoop 的创建是发生在第一次获取时,RunLoop 的销毁是发生在线程结束时。你只能在一个线程的内部获取其 RunLoop(主线程除外)。
四.注意事项
我们的RunLoop要想工作,必须要让它存在一个Item(source,observer或者timer),主线程之所以能够一直存在,并且随时准备被唤醒就是应为系统为其添加了很多Item
当perform selector在后台线程中执行的时候,这个线程必须有一个开启的runLoop
当scroll滑动时,同页面上的定时器为什么会暂停?
创建一个timer时,系统默认把timer添加到kCFRunLoopDefaultMode模式中,但是当页面滚动的时候,RunLoop的Mode会自动切换成UITrackingRunLoopMode模式,因此timer失效,当停止滑动,RunLoop又会切换回NSDefaultRunLoopMode模式,因此timer又会重新启动了。
怎么在tableview滑动时延迟加载图片来提高流畅度?
把加载图片放入NSDefaultRunLoopMode模式种就可以避免来
[self.imageView performSelector:@selector(setImage:) withObject:[UIImage imageNamed:@”abc”] afterDelay:2.0 inModes:@[NSDefaultRunLoopMode]];
常说的AFNetworking常驻线程保活是什么原理?
(void)networkRequestThreadEntryPoint:(id)__unused object {
@autoreleasepool {
[NSThread currentThread] setName:@”AFNetworking”];
NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
[runLoop addPort:[NSMachPort port] forMode:NSDefaultRunLoopMode];
[runLoop run];
}
}
(NSThread *)networkRequestThread {
static NSThread *_networkRequestThread = nil;
static dispatch_once_t oncePredicate;
dispatch_once(&oncePredicate, ^{
_networkRequestThread = [[NSThread alloc] initWithTarget:self selector:@selector(networkRequestThreadEntryPoint:) object:nil];
[_networkRequestThread start];
});
return _networkRequestThread;
}