##少用NSTimer
在iOS中开发中,定时器是经常用到的工具。一般情况下,我们都会去使用NSTimer
这个类。然而,这个类一旦使用不当的话,会给我们造成很多无法预见的坑。
NSTimer
有哪些坑,一个个来数一数
Timer必须作用在一个运行中的runloop
用过NSTimer
的都知道,在创建了NSTimer
之后,必须将NSTimer加入到runloop中。如果是在主线程的还好,因为主线程默认是有runLoop的(实际上一般很少会在主线程开timer)。如果是在子线程,那么就要手动激活runLoop先,不然就调用时无效的。
操作必须在同一个线程中
NSTimer的创建与撤销必须在同一个线程操作、performSelector的创建与撤销必须在同一个线程操作
内存泄漏
一般用NSTimer最容易出现的问题。首先来分析为什么会出现这个情况。
通常,我们创建一个NSTimer
的方法是调用以下的API:
1 | + (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo; |
这个时候,target会被timer持有,引用计数+1。
此时,如果直接销毁target,即使在target的deallco里面调用了[timer invaild]也无用。因为timer和target相关,因此必须提前调用[timer invaild]。这种BUG最常见的的就是一个UIViewController
pop的时候,timer没有被invaild,结果UIViewController
没有被销毁,导致了内存泄漏。
This method is the only way to remove a timer from an NSRunLoop object. The NSRunLoop object removes its strong reference to the timer, either just before the invalidate method returns or at some later point.
If it was configured with target and user info objects, the receiver removes its strong references to those objects as well.
You must send this message from the thread on which the timer was installed. If you send this message from another thread, the input source associated with the timer may not be removed from its run loop, which could prevent the thread from exiting properly.
注意到官方文档中提到,runLoop会持有NSTimer的引用,因此除了调用invaild
方法外,没有别的方法.
##用GCD代替NSTimer
用过GCD的都说好,可以说把GCD玩的很溜的话,很多异步并行问题都能解决,在这里我们就要去解决NSTimer的坑的问题。这里用三个开源库来抛砖引玉,讲讲不同的实现由什么优缺点。
###RNTimer
这个库很老了,12年后就没有更新了,内容也很简单,只有两个文件,因为代码很少,直接上源码了
源码摘要
1 |
|
实现分析
实现很简单,用的是最基本的方法,即创建了一个dispatch_source_t
,传入block
,然后通过dispatch_source_set_timer
将timer添加到dispatch_source
,设置事件处理为传入的block,然后执行timer。
提供两个和NSTimer
一样的方法,invalidate
和fire
。fire
会立即执行block,但是不会使timer失效,依然会按照预定计划执行。
在API定义上,保持了和NSTimer
一样的定义,所以切换起来没有压力。但是简单也以为这功能不完善。
- 无法暂停挂起timer
- 无法指定线程
- 无法添加新的任务到timer中
MSWeakTimer
MSWeakTimer
的API设计成和NSTimer
一样,调用的方法也一致,直接使用并没有什么副作用。
1 |
|
实现
先来看初始化方法:
1 | - (id)initWithTimeInterval:(NSTimeInterval)timeInterval |
将传入的参数复制给当前类的属性,然后初始化privateSerialQueue
,privateSerialQueue
的名称是privateQueueName
,可变参数为当前对象的内存地址。调用dispatch_set_target_queue
将传参的dispatchQueue
优先级赋给privateSerialQueue
。最后,初始化timer
。
然后看执行函数
1 | - (void)schedule |
首先调用了resetTimerProperties
1 | - (void)resetTimerProperties |
resetTimerProperties
重置timer
。
intervalInNanoseconds
和toleranceInNanoseconds
充当interval
和leeway
在执行完resetTimerProperties
后,dispatch_source_set_event_handler
一个block到timer
,执行timerFired
:
1 | - (void)timerFired |
这里OSAtomicAnd32OrigBarrier
是个值得关注的函数。
OSAtomicAnd32OrigBarrier
是一个原子操作的布尔与运算,且带有内存屏障。OSAtomicAnd32OrigBarrier
保证了在与操作之前数据结构的存储结构发生改变。
详细的只是可以参考下面这篇文章
http://southpeak.github.io/blog/2014/10/17/osatomicyuan-zi-cao-zuo/
在这里,_timerFlags.timerIsInvalidated
与1做与操作,如果返回return 则返回,如果不是,执行.
这里如果不加入invalidate
的话,恐怕说不清楚为什么这里要做这个判断
1 | - (void)invalidate |
1 | if (!OSAtomicTestAndSetBarrier(7, &_timerFlags.timerIsInvalidated)) |
OSAtomicTestAndSetBarrier
和OSAtomicAnd32OrigBarrier
正好可以作为一个相对操作。
OSAtomicTestAndSetBarrier(7, &_timerFlags.timerIsInvalidated)
会指定变量中的一个bit,把它设置为‘1’并返回旧值。如果要指定32bit integer最低位,n应该是7。因此如果第一次调用OSAtomicTestAndSetBarrier
的话,返回false同时话将timerIsInvalidated
置成1.由于原子操作和内存屏障,这个方法不会和OSAtomicAnd32OrigBarrier
一同执行。如果执行到timerFired
的时候,OSAtomicAnd32OrigBarrier(1, &_timerFlags.timerIsInvalidated)
返回true,就会返回了。通过这个方法保证了在调用invalidate
后将timer失效。
timerFired
如果能执行,则通过performSelector
来调用对用的SEL
在invalidate
中,如果需要释放,需要调用dispatch_async
。原因是我们虽然通过原子操作保证了invalidated
,但是我们无法确定当前的上下文情况,使用dispatch_async
可能会造成死锁。
主流程分析就是如此,MSWeakTimer
最值得关注的就是上面讲解决线程同步的方法。
MSWeakTimer
解决了前面提到的问题中无法添加指定线程(实际上也不是添加到那个线程,而是新开了一个线程)。由于模仿的是NSTimer
的API,所以另外两个方法也没有解决。
我的实现 – ZXGCDTimer
在参考了上面的两个开源库后,自己动手撸了一个简单的基于GCD的Timer。基本解决了上面提到了三个问题。
代码地址:https://github.com/csbzhixing/ZXGCDTimer
思路是,通过ZXGCDTimerManager
提供对外操作包括创建,执行,取消,挂起,回复功能。每个timer通过ZXGCDTimer
为单位去管理,以timerName
去区分不同的timer。支持自定义线程,支持新增Action。
由于是自己处于研究目的写的,可能存在各种问题,也欢迎大家提出批评和指正。