iOS —— NSRunLoop(一)

一、NSRunLoop

+ (NSRunLoop *)currentRunLoop
    如果调用的线程中没有runloop,那么将会创建一个并返回
  + (NSRunLoop *)mainRunLoop
    返回主线程的runloop

  - (void)acceptInputForMode:(NSString *)mode beforeDate:(NSDate *)limitDate
    运行loop一次或者直到limitDate。如果没有input sources加入到这个loop,那么马上返回;否则一直运行到limitDate,或者接口到一个input source然后返回。
  - (void)addPort:(NSPort *)aPort forMode:(NSString *)mode
  - (void)addTimer:(NSTimer *)aTimer forMode:(NSString *)mode
    port和timer都可以添加到多个mode中
  - (void)cancelPerformSelector:(SEL)aSelector target:(id)target argument:(id)anArgument
    取消所有mode中的perform select,argument必须跟指定调用时候的一样
  - (void)cancelPerformSelectorsWithTarget:(id)target
  - (NSString *)currentMode
    如果run loop没有运行,那么返回nil
  - (CFRunLoopRef)getCFRunLoop
  - (NSDate *)limitDateForMode:(NSString *)mode
    下一次运行的时间,如果没有指定的mode上没有input source,返回nil
  - (void)performSelector:(SEL)aSelector target:(id)target argument:(id)anArgument order:(NSUInteger)order modes:(NSArray *)modes
order值越低优先级越高
  - (void)removePort:(NSPort *)aPort forMode:(NSString *)mode
  - (void)run
    在default mode下无限运行loop,但是如果没有任何input source,会立即返回。手动移除所有已知的inout source并不能保证run loop停止运行,因为系统可能会添加一些input source。
  - (BOOL)runMode:(NSString *)mode beforeDate:(NSDate *)limitDate
    运行input source一次,为指定mode的input阻塞直到date的时间。如过没有input source,立即返回并返回NO。
  - (void)runUntilDate:(NSDate *)limitDate
  如果没有input source,立即返回。否则在limitDate到来之前,不停的循环。
再详细的就看文档吧

NSRunLoop是消息机制的处理模式。


NSRunLoop的作用在于有事情做的时候使的当前NSRunLoop的线程工作,没有事情做让当前NSRunLoop的线程休眠。

NSRunLoop就是一直在循环检测,从线程start到线程end,检测inputsource(如点击,双击等操作)同步事件,检测timesource同步事件,检测到输入源会执行处理函数,首先会产生通知,corefunction向线程添加runloop observers来监听事件,意在监听事件发生时来做处理。

在单线程的app中,不需要注意Run Loop,但不代表没有。程序启动时,系统已经在主线程中加入了Run Loop。它保证了我们的主线程在运行起来后,就处于一种“等待”的状态(而不像一些命令行程序一样运行一次就结束了),这个时候如果有接收到的事件(Timer的定时到了或是其他线程的消息),就会执行任务,否则就处于休眠状态。


二、RunLoopMode

NSDefaultRunLoopMode 这是大多数操作中使用的模式。
NSConnectionReplyMode	该模式用来监控NSConnection对象。你通常不需要在你的代码中使用该模式。
NSModalPanelRunLoopMode	Cocoa使用该模式来标识用于modal panel(模态面板)的事件。
NSEventTracking(UITrackingRunLoopMode) Cocoa使用该模式来处理用户界面相关的事件。
NSRunLoopCommonModes 这是一组可配置的通用模式。将input sources与该模式关联则同时也将input sources与该组中的其它模式进行了关联。对于Cocoa应用,该模式缺省的包含了default,modal以及event tracking模式。

  一个常见的问题就是,主线程中一个NSTimer添加在default mode中,当界面上有一些scroll view的滚动频繁发生导致run loop运行在UItraking mode中,从而这个timer没能如期望那般的运行。所以,我们就可以把这个timer加到NSRunLoopCommonModes中来解决(iOS中)。


三、Loop


其中Input source是一些异步的事件,比如port,selector等,这个会让runUntilDate:跳出(当然指的是非主线程中的runloop)。Timer source是同步的,一个timer结束后,在重复时间后或者手动fire后才会再一次调用。

在来看看这张图片

它说明了用户对UI的操作实际上是一种port,会放到一个队列中传到loop,然后由loop交给主线程处理。loop就是一个循环,接受event,传递,继续。主线程是另一个循环,负责事件的处理与界面的显示。当然这两者关系复杂。

在看下面的代码

复制代码
BOOL pageStillLoading = YES;
-(void)press:(id)sender
{
    [(UIButton*)sender setSelected:YES];
    NSLog(@"begin"); // 1
    [[NSRunLoop mainRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:10.0]];  // 2
    NSLog(@"end"); // 3

    pageStillLoading = YES;
    [NSThread detachNewThreadSelector:@selector(loadPageInBackground:)toTarget:self withObject:nil]; // 4
    while (pageStillLoading) {
        [[NSRunLoop currentRunLoop] runMode:UITrackingRunLoopMode beforeDate:[NSDate distantFuture]]; // 5
        NSLog(@"while end");
    }
    NSLog(@"over");
    
}

-(void)loadPageInBackground:(id)sender
{
    sleep(3); // 6
    NSLog(@"timer"); // 7
    pageStillLoading = NO; // 8
}
复制代码

 

我在viewcontroller的view上加了一个uibutton,并且事件是press。touch up inside,然后,不在碰触界面(A)。发生了什么?
看看log:

2013-01-06 00:57:21.167 runloop[10146:c07] begin
2013-01-06 00:57:31.171 runloop[10146:c07] end
2013-01-06 00:57:34.173 runloop[10146:3703] timer
2013-01-06 00:58:00.001 runloop[10146:c07] while end
2013-01-06 00:58:00.002 runloop[10146:c07] over


pageStillLoading设置成NO之后过了近30s,while才结束。
如果我把5的mode改成NSRunLoopCommonModes或者在界面上在加一个按钮,然后不停的点击那个按钮(B),结果如下

复制代码
2013-01-06 01:01:23.944 runloop[10174:c07] begin
2013-01-06 01:01:33.948 runloop[10174:c07] end
2013-01-06 01:01:34.023 runloop[10174:c07] while end
.
.
.
2013-01-06 01:01:36.943 runloop[10174:c07] while end
2013-01-06 01:01:36.950 runloop[10174:370b] timer
2013-01-06 01:01:37.016 runloop[10174:c07] while end
2013-01-06 01:01:37.016 runloop[10174:c07] over
复制代码

 

为什么?我们看下流程
当2执行的时候,主线程的事情就是运行runloop 10s,而这个runloop本来就在运行的,所以相当于主线程什么都不用做,空闲状态的主线程当然也可以继续响应界面上的ui事件哦。
当5执行的时候,也没有主线程什么事情,runMode:beforeDate:是要么接受到一个mode上的event,要么到date这个时间。所以,A中要等待很久,这句才会返回;而B中,却是不停的给他event,所以while语句执行了多次。



参考:http://www.cnblog.com/v2m_/archive/2013/01/06/2846984.html





  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值