菜鳥(niǎo)教程——NSTimer使用

app中有一個(gè)計(jì)時(shí)功能衙荐。之前使用了簡(jiǎn)單的在主線(xiàn)程中調(diào)用:

?+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(id)userInfo repeats:(BOOL)yesOrNo;

但是當(dāng)每0.01秒進(jìn)行一次repeat操作時(shí)捞挥,NSTimer是不準(zhǔn)的,嚴(yán)重滯后赫模,而改成0.1秒repeat操作树肃,則這種滯后要好一些蒸矛。

導(dǎo)致誤差的原因是我在使用“scheduledTimerWithTimeInterval”方法時(shí)瀑罗,NSTimer實(shí)例是被加到當(dāng)前runloop中的,模式是NSDefaultRunLoopMode雏掠。而“當(dāng)前runloop”就是應(yīng)用程序的main runloop斩祭,此main runloop負(fù)責(zé)了所有的主線(xiàn)程事件,這其中包括了UI界面的各種事件乡话。當(dāng)主線(xiàn)程中進(jìn)行復(fù)雜的運(yùn)算摧玫,或者進(jìn)行UI界面操作時(shí),由于在main runloop中NSTimer是同步交付的被“阻塞”,而模式也有可能會(huì)改變诬像。因此屋群,就會(huì)導(dǎo)致NSTimer計(jì)時(shí)出現(xiàn)延誤。

解決這種誤差的方法坏挠,一種是在子線(xiàn)程中進(jìn)行NSTimer的操作芍躏,再在主線(xiàn)程中修改UI界面顯示操作結(jié)果;另一種是仍然在主線(xiàn)程中進(jìn)行NSTimer操作降狠,但是將NSTimer實(shí)例加到main runloop的特定mode(模式)中对竣。避免被復(fù)雜運(yùn)算操作或者UI界面刷新所干擾。

方法一:

在開(kāi)始計(jì)時(shí)的地方:

?if (self.timer) {

? ? ? ? [self.timer invalidate];

? ? ? ? ?self.timer = nil;

? ? ?}

? ? ?self.timer = [NSTimer timerWithTimeInterval:0.01 target:self selector:@selector(addTime) userInfo:nil repeats:YES];

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

[NSRunLoop currentRunLoop]獲取的就是“main runloop”榜配,使用NSRunLoopCommonModes模式否纬,將NSTimer加入其中。

方法二:

開(kāi)辟子線(xiàn)程:(使用子線(xiàn)程的runloop)

?NSThread *thread = [[NSThread alloc] initWithTarget:self selector:@selector(newThread) object:nil];

? ? ?[thread start];

- (void)newThread

?{

? ? @autoreleasepool

? ? {

? ? ? ?[NSTimer scheduledTimerWithTimeInterval:2.0 target:self selector:@selector(addTime) userInfo:nil repeats:YES];

? ? ? ?[[NSRunLoop currentRunLoop] run];? ? ?}

?}

在子線(xiàn)程中將NSTimer以默認(rèn)方式加到該線(xiàn)程的runloop中蛋褥,啟動(dòng)子線(xiàn)程临燃。

方法三:

使用GCD,同樣也是多線(xiàn)程方式:

聲明全局成員變量

dispatch_source_t _timers;

? ? ?uint64_t interval = 0.01 * NSEC_PER_SEC;

? ? dispatch_queue_t queue = dispatch_queue_create("my queue", 0);

? ? ?_timers = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);

? ? ?dispatch_source_set_timer(_timers, dispatch_time(DISPATCH_TIME_NOW, 0), interval, 0);

? ? ?__weak ViewController *blockSelf = self;

? ? ?dispatch_source_set_event_handler(_timers, ^()

? ? {

NSLog(@"Timer %@", [NSThread currentThread]);

? ? ? ?[blockSelf addTime];

? ?});

? ? ?dispatch_resume(_timers);

然后在主線(xiàn)程中修改UI界面:

?dispatch_async(dispatch_get_main_queue(), ^{

? ? ? ? self.label.text = [NSString stringWithFormat:@"%.2f", self.timeCount/100];

? ? ?});

runloop是一個(gè)看似很神秘的東西壁拉,其實(shí)一點(diǎn)也不神秘谬俄。每個(gè)線(xiàn)程都有一個(gè)實(shí)際已經(jīng)存在的runloop。比如我們的主線(xiàn)程弃理,在主函數(shù)的UIApplication中:

?UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]))

系統(tǒng)就為我們將主線(xiàn)程的main runloop隱式的啟動(dòng)了溃论。runloop顧名思義就是一個(gè)“循環(huán)”,他不停地運(yùn)行痘昌,從程序開(kāi)始到程序退出钥勋。正是由于這個(gè)“循環(huán)”在不斷地監(jiān)聽(tīng)各種事件,程序才有能力檢測(cè)到用戶(hù)的各種觸摸交互辆苔、網(wǎng)絡(luò)返回的數(shù)據(jù)才會(huì)被檢測(cè)到算灸、定時(shí)器才會(huì)在預(yù)定的時(shí)間觸發(fā)操作……

runloop只接受兩種任務(wù):輸入源和定時(shí)源。本文中說(shuō)的就是定時(shí)源驻啤。默認(rèn)狀態(tài)下菲驴,子線(xiàn)程的runloop中沒(méi)有加入我們自己的源,那么我們?cè)谧泳€(xiàn)程中使用自己的定時(shí)器時(shí)骑冗,就需要自己加到runloop中赊瞬,并啟動(dòng)該子線(xiàn)程的runloop,這樣才能正確的運(yùn)行定時(shí)器贼涩。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末巧涧,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子遥倦,更是在濱河造成了極大的恐慌谤绳,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,686評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異缩筛,居然都是意外死亡消略,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,668評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén)瞎抛,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)疑俭,“玉大人,你說(shuō)我怎么就攤上這事婿失〕В” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 158,160評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵豪硅,是天一觀的道長(zhǎng)哩照。 經(jīng)常有香客問(wèn)我,道長(zhǎng)懒浮,這世上最難降的妖魔是什么飘弧? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,736評(píng)論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮砚著,結(jié)果婚禮上次伶,老公的妹妹穿的比我還像新娘。我一直安慰自己稽穆,他們只是感情好冠王,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,847評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著舌镶,像睡著了一般柱彻。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上餐胀,一...
    開(kāi)封第一講書(shū)人閱讀 50,043評(píng)論 1 291
  • 那天哟楷,我揣著相機(jī)與錄音,去河邊找鬼否灾。 笑死卖擅,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的墨技。 我是一名探鬼主播惩阶,決...
    沈念sama閱讀 39,129評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼健提!你這毒婦竟也來(lái)了琳猫?” 一聲冷哼從身側(cè)響起伟叛,我...
    開(kāi)封第一講書(shū)人閱讀 37,872評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤私痹,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體紊遵,經(jīng)...
    沈念sama閱讀 44,318評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡账千,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,645評(píng)論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了暗膜。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片匀奏。...
    茶點(diǎn)故事閱讀 38,777評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖学搜,靈堂內(nèi)的尸體忽然破棺而出娃善,到底是詐尸還是另有隱情,我是刑警寧澤瑞佩,帶...
    沈念sama閱讀 34,470評(píng)論 4 333
  • 正文 年R本政府宣布聚磺,位于F島的核電站,受9級(jí)特大地震影響炬丸,放射性物質(zhì)發(fā)生泄漏瘫寝。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,126評(píng)論 3 317
  • 文/蒙蒙 一稠炬、第九天 我趴在偏房一處隱蔽的房頂上張望焕阿。 院中可真熱鬧,春花似錦首启、人聲如沸暮屡。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,861評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)栽惶。三九已至,卻和暖如春疾嗅,著一層夾襖步出監(jiān)牢的瞬間外厂,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,095評(píng)論 1 267
  • 我被黑心中介騙來(lái)泰國(guó)打工代承, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留汁蝶,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,589評(píng)論 2 362
  • 正文 我出身青樓论悴,卻偏偏與公主長(zhǎng)得像掖棉,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子膀估,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,687評(píng)論 2 351

推薦閱讀更多精彩內(nèi)容

  • 原文地址:http://blog.ibireme.com/2015/05/18/runloop/ RunLoop ...
    大餅炒雞蛋閱讀 1,152評(píng)論 0 6
  • 轉(zhuǎn)載:http://www.cocoachina.com/ios/20150601/11970.html RunL...
    Gatling閱讀 1,436評(píng)論 0 13
  • 深入理解RunLoop 由ibireme| 2015-05-18 |iOS,技術(shù) RunLoop 是 iOS 和 ...
    橙娃閱讀 849評(píng)論 1 2
  • 轉(zhuǎn)自http://blog.ibireme.com/2015/05/18/runloop 深入理解RunLoop ...
    飄金閱讀 976評(píng)論 0 4
  • 第一次知道后海大鯊魚(yú)的名字幔亥,是因?yàn)楹篚彺蛉耸录N乙虼藢?duì)后鯊沒(méi)有什么好印象察纯。 后來(lái)聽(tīng)歌的時(shí)候帕棉,無(wú)意中聽(tīng)到后鯊的「猛...
    社長(zhǎng)墨非閱讀 567評(píng)論 0 0