初探NSRunLoop

RunLoop 简介

RunLoop 接收输入事件有两种不同的源:输入源和定时源。输入源传递异步事件,通常消息来自其他线程和程序。定时源则传递同步事件,发生在重复的时间或者重复的时间间隔。


1.runloop 初探

runloop,顾名思义,是一个循环。有事件去处理的时候就去处理,没事件处理的时候就休息。

本文结合NSTimer来初步学习runloop

场景    在runloop的默认模式下添加一个timer,然后加一个UI控件 

NSTimer*timer = [NSTimerscheduledTimerWithTimeInterval:1.0target:selfselector:@selector(timerMethod)userInfo:nilrepeats:YES];

- (void)timerMethod

{

staticint num =0;

NSLog(@"%@ %d",[NSThreadcurrentThread],num++);

}


运行后,timer每秒打印一次,打印可以知道timer加在了主线程。担当拖动UI控件的时候,timer停止打印了。原因并不是阻塞了主线程(如果阻塞了主线程,那么UI控件也不会动了)。真正原因是:timer执行的时候,runloop在默认模式下执行timer。拖动界面的时候(source源),runloop在UI模式下去执行UI事件,拖住控件不松手,runloop就会一直处理UI事件,不再去处理timer源事件。

runloop的两种重要模式:

(1)NSDefaultRunLoopMode runloop的默认模式,只要有事件就处理

(2)UITrackingRunLoopMode (优先切换)这个模式,是在有UI事件的时候切换到的模式

备注:NSRunLoopCommonModes  占位符,不算是一种模式(默认模式和UI模式的结合)

尝试解决办法1:把timer放在UI模式下

NSTimer *timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timerMethod) userInfo:nil repeats:YES];                                         [[NSRunLoop currentRunLoop]addTimer:timer1 forMode:NSRunLoopCommonModes];

此时再运行,发现每秒打印一次没问题。貌似解决了问题。但是,并不完美。

如果timerMethod里面做了耗时操作,会有什么结果呢?修改timerMethod方法如下

- (void)timerMethod

{

[NSThread sleepForTimeInterval:1.0];

staticint num =0;

NSLog(@"%@ %d",[NSThreadcurrentThread],num++);

}

这是再运行,我们会发现,即使拖动UI控件也没问题,每秒会打印一次。但是由于有耗时操作,由于耗时操作是在主线程,所以耗时操作的时候UI控件会卡顿。

尝试解决方案2   把timer放在子线程

NSThread *thread = [[NSThread alloc]initWithBlock:^{

    NSTimer *timer1 = [NSTimer timerWithTimeInterval:1.0 target:self        selector:@selector(timerMethod) userInfo:nil repeats:YES];

    [[NSRunLoop currentRunLoop]addTimer:timer1 forMode:NSRunLoopCommonModes];

}];

    [thread start];

思考:timerMethod会执行吗?

运行后发现:timerMethod并没有被执行。原因是子线程,执行完任务被回收,所以不会执行timerMethod方法。子线程被回收是因为,子线程的runloop默认是不开启的。

下面做如下修改

NSThread *thread = [[NSThread alloc]initWithBlock:^{

    NSTimer *timer1 = [NSTimer timerWithTimeInterval:1.0 target:self     selector:@selector(timerMethod) userInfo:nil repeats:YES];

    [[NSRunLoop currentRunLoop]addTimer:timer1 forMode:NSRunLoopCommonModes];

    [[NSRunLoop currentRunLoop]run];

    NSLog(@"timer初始化%@",[NSThread currentThread]); //这个会执行吗????

}];

[thread start];

猜想:打印timer初始化的那句代码会被执行吗?

答案是否定的。因为[[NSRunLoop currentRunLoop]run]; 这句代码是一个死循环。进入处理事件循环,如果没有事件则立刻返回,因为一直有timer事件,所以一直无法返回。

尝试解决方案3  自己管理runloop循环

_finished 为属性 初始值为yes

- (void)runUntilDate:(NSDate*)limitDate;//同run方法,增加超时参数limitDate,避免进入无限循环。

NSThread*thread = [[NSThreadalloc]initWithBlock:^{

    NSTimer*timer1 =      [NSTimer     timerWithTimeInterval:1.0 target:self selector:@selector(timerMethod) userInfo:nil repeats:YES];

[[NSRunLoop currentRunLoop]addTimer:timer1 forMode:NSRunLoopCommonModes];

while (_finished) {

[[NSRunLoop currentRunLoop]runUntilDate:[NSDate dateWithTimeIntervalSinceReferenceDate:0.01]];// 每0.01秒开始一个runloop,超过0.01秒。runloop自动退出。

}

NSLog(@"timer初始化%@",[NSThread currentThread]);//这个会执行吗???

}];

[thread start];

NSLog(@"main thread");//这句主线程运行的,可以执行

}

为了使runloop退出  需要在适当的地方修改finished为no,这样就可以退出while循环

源码:https://github.com/mhlee0514/RunLoop

参考资料:http://www.cnblogs.com/tangbinblog/archive/2012/12/07/2807088.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,440评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,814评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,427评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,710评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,625评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,014评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,511评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,162评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,311评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,262评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,278评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,989评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,583评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,664评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,904评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,274评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,856评论 2 339

推荐阅读更多精彩内容

  • 基本概念 进程 进程是指在系统中正在运行的一个应用程序,而且每个进程之间是独立的,它们都运行在其专用且受保护的内存...
    小枫123阅读 878评论 0 1
  • 一、什么是runloop 字面意思是“消息循环、运行循环”。它不是线程,但它和线程息息相关。一般来讲,一个线程一次...
    WeiHing阅读 8,104评论 11 111
  • RunLoop 文章目录 RunLoop简介 1.1 什么是RunLoop? 1.2 RunLoop和线程 1.3...
    May_d8f1阅读 280评论 0 1
  • 原文地址:http://blog.ibireme.com/2015/05/18/runloop/ RunLoop ...
    大饼炒鸡蛋阅读 1,140评论 0 6
  • Runloop是iOS和OSX开发中非常基础的一个概念,从概念开始学习。 RunLoop的概念 -般说,一个线程一...
    小猫仔阅读 976评论 0 1