创建一个NSTimer,有多种方法,如
scheduledTimerWithTimeInterval:target:selector:userInfo:repeats:
timerWithTimeInterval:target:selector:userInfo:repeats:
等等
其中,用不带有scheduled
的,在初始化了后,还需要调用
addTimer:forMode:
方法,将Timer加入到runloop中。
参数:
-
seconds
定时时间 -
target
发送对象 -
aSelector
定时器启动调用的方法 -
userInfo
可为nil,使用者的信息 -
repeats
定时器是否循环
在Timer添加到runloop后,定时器立刻启动,在seconds时间后,调用aSelector
方法,如果最开始时不想等seconds时间,可调用-fire方法,立刻调用aSelector
方法。
-invalidate
方法可使定时器失效。
注意:
- 使用
-invalidate
方法后,即使再次使用-fire方法也不会让定时器再次工作,因为-invalidate
方法是使Timer失效。 - 暂停定时器:可使用
[self setFireDate:[NSDate distantFuture]];
方法:即将定时器启动时间设为无穷远
恢复定时器,使用[self setFireDate:[NSDate date]];
方法:即将定时器启动时间设为现在的时刻。 - 对于需要在暂停时刻改变定时时间间隔的情况,则需要重新创建NSTImer,启动,通过-invalidate停止定时器。
NSTimer 内存泄漏解决方法
由于 NSTimer
容易与持有它的对象互相引用,从而导致内存泄漏,因此在使用时需要格外注意。 iOS 10.0 以后已经有了 scheduledTimer(withTimeInterval:repeats:block:)
方法解决内存泄漏问题,但是 iOS 10.0 以前还需要自己想办法。
- 如果
timerWithTimeInterval:target:selector:userInfo:repeats:
中传给 target 的为weakSelf
是否可以?
不可以。因为 Timer 对 target 强引用,强引用一个 weak 变量,对变量所指的对象仍然是强引用 - 借鉴参考 1 中第 52 条的方法解决该问题。
extension Timer {
class func njyScheduledTimer(timerInterval: TimeInterval, block: @escaping () -> Void, repeats: Bool) -> Timer {
return Timer.scheduledTimer(timeInterval: timerInterval, target: self, selector: #selector(njyBlockInvoke(timer:)), userInfo: [block copy], repeats: repeats)
}
@objc class func njyBlockInvoke(timer: Timer) {
let block = timer.userInfo as? () -> Void
if let block = block {
block()
}
}
}
注意:在实际使用时,仍然要注意内存泄漏的问题,因此需要使用 weak var weakSelf = self
和 let strongSelf = weakSelf
打破保留环,解决内存泄漏的问题。