定时器NSTimer用法

创建一个NSTimer,有多种方法,如

scheduledTimerWithTimeInterval:target:selector:userInfo:repeats:
timerWithTimeInterval:target:selector:userInfo:repeats:

等等
其中,用不带有scheduled的,在初始化了后,还需要调用
addTimer:forMode:方法,将Timer加入到runloop中。


参数:

  1. seconds定时时间
  2. target发送对象
  3. aSelector定时器启动调用的方法
  4. userInfo可为nil,使用者的信息
  5. 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 以前还需要自己想办法。

  1. 如果 timerWithTimeInterval:target:selector:userInfo:repeats: 中传给 target 的为 weakSelf 是否可以?
    不可以。因为 Timer 对 target 强引用,强引用一个 weak 变量,对变量所指的对象仍然是强引用
  2. 借鉴参考 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 = selflet strongSelf = weakSelf 打破保留环,解决内存泄漏的问题。

参考

  1. Effective Objective-C 2.0

你可能感兴趣的:(定时器NSTimer用法)