最近在开发中计划实现一个滚动新闻(TableView的其中cell中实现),于是我定义了一个NSTimer定时器开实现一直滚动,但是当我发现了当我拖懂TableView时候,滚动新闻就停止了.
于是我首先想到的是用GCD创建一个异步线程,将定时器放到这里,为了让定时器更准确,我在定时器后面添加了[[NSRunLoop currentRunLoop] run];(因为定时器在子线程中默认是不启动的) 之后确实实现了拖动tableView的时候,滚动新闻也滚动,但是dang当进入别的页面在返回当前页面的时候,有时会发生内存错误
-[UIViewAnimationState class]: message sent to deallocated instance 0x7fc9ec351490
在终端定位找到
rollNewCell timer] | -[NSRunLoop(NSRunLoop) run] | -[NSRunLoop(NSRunLoop) runMode:beforeDate:] | CFRunLoopRunSpecific | __CFRunLoopRun | __CFRunLoopDoTimer | __CFRUNLOOP_IS_CALLING_OUT_TO_A_TIMER_CALLBACK_FUNCTION__ | __NSFireTimer | -[SQ_ScrollNewCell timerChanged:] |
我才发现是之前nstimer定时器的问题
经过查阅资料和别人帮助终于找到了解决方案
直接在主线程设置NSTimer
self.timer = [NSTimer scheduledTimerWithTimeInterval:0.2 target:self selector:@selector(timerChanged:) userInfo:nil repeats:YES];
[[NSRunLoop currentRunLoop] addTimer:_timer forMode:NSRunLoopCommonModes];
//:NSRunLoopCommonModes,这个模式等效于NSDefaultRunLoopMode和NSEventTrackingRunLoopMode的结合。
结论
当系统启动时会默认创建一个runloop模式是NSDefaultRunLoopMode,当使用NSTimer的scheduledTimerWithTimeInterval方法时。事实上此时Timer会被加入到当前线程的RunLoop中,且模式是默认的NSDefaultRunLoopMode。而如果当前线程就是主线程,也就是UI线程时,某些UI事件,比如UIScrollView的拖动操作,会将Run Loop切换成NSEventTrackingRunLoopMode模式,在这个过程中,默认的NSDefaultRunLoopMode模式中注册的事件是不会被执行的。也就是说,此时使用scheduledTimerWithTimeInterval添加到Run Loop中的Timer就不会执行。
所以为了设置一个不被UI干扰的Timer,我们需要手动创建一个Timer,然后使用NSRunLoop的addTimer:forMode:方法来把Timer按照指定模式加入到Run Loop中。这里使用的模式是:NSRunLoopCommonModes,这个模式等效于NSDefaultRunLoopMode和NSEventTrackingRunLoopMode的结合。
这个问题开启了我对runloop的学习的兴趣