RAC 與 NSTimer

NSTimer時(shí)鐘事件:

一般情況 我們會(huì)這樣擼

[NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timeMethod) userInfo:nil repeats:YES];

- (void)timeMethod{

? ? NSLog(@"timeMethod來(lái)了");

}

//這個(gè)時(shí)候 timeMethod來(lái)了每秒鐘會(huì)打印一次

好問(wèn)題來(lái)了,這個(gè) 時(shí)候我拖一個(gè)UITextView放到控制器许蓖,當(dāng)我們一直拖動(dòng)UITextView的時(shí)候,“ timeMethod來(lái)了每秒鐘會(huì)打印一次”是不是停下來(lái)了。

為什么乱凿?

--首先我們要搞清楚因块,NSTimer事件是交給誰(shuí) 處理殴玛?--"Runloop"在處理,系統(tǒng)會(huì)把這個(gè)事件包裝成“souce”事件源給RunLoop處理束析;

那RunLoop在處理這些事件的時(shí)候有什么區(qū)分和區(qū)別呢?

--是有的憎亚,從下面兩個(gè)圖可以看出Runloop處理事件會(huì)有優(yōu)先級(jí)的员寇,UI模式級(jí)別最高弄慰,所有當(dāng)我們拖動(dòng)UITextView的時(shí)候,RunLoop在處理UITextView的觸摸事件(這個(gè)事件是UI模式下的事件 )蝶锋,timeMethod來(lái)了沒(méi)有打印了


RunLoop


UI模式和默認(rèn)模式

好現(xiàn)在我們來(lái)做個(gè)試驗(yàn)陆爽,假設(shè)我們想要在觸摸UI 的時(shí)候,Timer事件還是會(huì)被處理扳缕,那我們添加到RunLoopd的UI模式下

??? NSTimer *timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timeMethod) userInfo:nil repeats:YES];
???
??? //UITrackingRunLoopMode
??? [[NSRunLoop currentRunLoop] addTimer:timer forMode:UITrackingRunLoopMode];

奇怪了慌闭,這個(gè)時(shí)候只有在觸摸UI的時(shí)候才會(huì)打印,

其實(shí):

NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timeMethod) userInfo:nil repeats:YES];

等同于下面的默認(rèn)模式躯舔,觸摸UI的時(shí)候RunLoop不會(huì)處理UI模式下的事件

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

? ? //UITrackingRunLoopMode

? ? [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];

如果我們想普通模式和UI模式下驴剔,都有事件響應(yīng),那邊我們需要把這個(gè)Timer添加到第3中模式粥庄,站位模式"NSRunLoopCommonModes"

//總共有5中模型 UI 默認(rèn) 占位模式(默認(rèn)&UI模式) 程序啟動(dòng)模式 內(nèi)核模式

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

? ? [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];


以為站位模式就萬(wàn)事大吉了沒(méi)仔拟,NO,避免我們這個(gè)Timer做一個(gè)耗時(shí)操作飒赃,運(yùn)行下面代碼(在timeMethod讓線程睡2秒)

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

? ? [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];


- (void)timeMethod{

? ? [NSThread sleepForTimeInterval:2.0];

? ? NSLog(@"timeMethod來(lái)了");

}

這個(gè)時(shí)候是不是拓動(dòng)UI的時(shí)候有卡頓現(xiàn)象利花。那么這種我們?nèi)绾翁幚聿挥绊慤I卡頓呢?

--此時(shí)此刻载佳,我們要記住每條線程都有一個(gè)RunLoop,默認(rèn)情況是不開(kāi)啟的炒事。

是不是現(xiàn)在又靈感有來(lái)了,我們把這個(gè)Timer丟到一個(gè)線程里面去

NSThread *thread =? [[NSThread alloc] initWithBlock:^{
??????????? NSTimer *timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timeMethod) userInfo:nil repeats:YES];
??????????? [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
?????????? [[NSRunLoop currentRunLoop] run];//重點(diǎn)蔫慧,RunLoop,默認(rèn)情況是不開(kāi)啟的挠乳。注意要啟動(dòng)
?????????? NSLog(@"come here!");//這里不會(huì)打印,Runloop是死循環(huán)
??? }];
???
??? [thread start];
?


- (void)timeMethod{
??? [NSThread sleepForTimeInterval:2.0];
??? NSLog(@"timeMethod來(lái)了%@",[NSThread currentThread]);
}

運(yùn)行上面代碼姑躲,這個(gè)時(shí)候是不是不會(huì)出現(xiàn)UI卡頓了睡扬。

問(wèn)題有來(lái)l,RunLoop是不是停不下來(lái)了,要停下來(lái)黍析,要怎么玩卖怜?請(qǐng)看下面


NSThread *thread =? [[NSThread alloc] initWithBlock:^{
??????????? NSTimer *timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timeMethod) userInfo:nil repeats:YES];
??????????? [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
?????? while (!_finished) {
?????????? [[NSRunLoop currentRunLoop] runUntilDate:[NSDate dateWithTimeIntervalSinceNow:0.001]];
?????? }
????????? // [[NSRunLoop currentRunLoop] run];
?????????? NSLog(@"come here! %@",[NSThread currentThread]);
??? }];
???
??? [thread start];
?
}

- (void)timeMethod{
??? NSLog(@"timeMethod來(lái)了%@",[NSThread currentThread]);
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
??? _finished = YES;
}

一旦RunLoop停止,come here就好打印出來(lái)阐枣, 線程執(zhí)行完就釋放了马靠。

其實(shí)我們還有種簡(jiǎn)單方式處理這種需求

/GCD timer 設(shè)置Time


//注意要強(qiáng)引用self.timer

? self.timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, dispatch_get_global_queue(0, 0));

? ? dispatch_source_set_timer(self.timer, DISPATCH_TIME_NOW, 2 * NSEC_PER_SEC, 0 * NSEC_PER_SEC);//2 開(kāi)始時(shí)間 、持續(xù)時(shí)間蔼两、

? ? dispatch_source_set_event_handler(self.timer, ^{

? ? ? ? NSLog(@"GCD timer:%@", [NSThread currentThread]); //打印的是在子線程

? ? });

? ? dispatch_resume(self.timer); //啟動(dòng)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末甩鳄,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子额划,更是在濱河造成了極大的恐慌妙啃,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,839評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件俊戳,死亡現(xiàn)場(chǎng)離奇詭異揖赴,居然都是意外死亡茁瘦,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門储笑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)甜熔,“玉大人,你說(shuō)我怎么就攤上這事突倍∏幌。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,116評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵羽历,是天一觀的道長(zhǎng)焊虏。 經(jīng)常有香客問(wèn)我,道長(zhǎng)秕磷,這世上最難降的妖魔是什么诵闭? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,371評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮澎嚣,結(jié)果婚禮上疏尿,老公的妹妹穿的比我還像新娘。我一直安慰自己易桃,他們只是感情好褥琐,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著晤郑,像睡著了一般敌呈。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上造寝,一...
    開(kāi)封第一講書(shū)人閱讀 49,111評(píng)論 1 285
  • 那天磕洪,我揣著相機(jī)與錄音,去河邊找鬼诫龙。 笑死析显,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的赐稽。 我是一名探鬼主播叫榕,決...
    沈念sama閱讀 38,416評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼浑侥,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼姊舵!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起寓落,我...
    開(kāi)封第一講書(shū)人閱讀 37,053評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤括丁,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后伶选,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體史飞,經(jīng)...
    沈念sama閱讀 43,558評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡尖昏,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了构资。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片抽诉。...
    茶點(diǎn)故事閱讀 38,117評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖吐绵,靈堂內(nèi)的尸體忽然破棺而出迹淌,到底是詐尸還是另有隱情,我是刑警寧澤己单,帶...
    沈念sama閱讀 33,756評(píng)論 4 324
  • 正文 年R本政府宣布唉窃,位于F島的核電站,受9級(jí)特大地震影響纹笼,放射性物質(zhì)發(fā)生泄漏纹份。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評(píng)論 3 307
  • 文/蒙蒙 一廷痘、第九天 我趴在偏房一處隱蔽的房頂上張望蔓涧。 院中可真熱鬧,春花似錦笋额、人聲如沸蠢笋。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,315評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)昨寞。三九已至,卻和暖如春厦滤,著一層夾襖步出監(jiān)牢的瞬間援岩,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,539評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工掏导, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留享怀,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,578評(píng)論 2 355
  • 正文 我出身青樓趟咆,卻偏偏與公主長(zhǎng)得像添瓷,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子值纱,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評(píng)論 2 345