从此
文章
📄文章 #️⃣专题 🌐上网 📺 🛒 📱

iOS开发的3种Timer定时器

🕗2023-05-11
在iOS中有3种常见的定时器,它们可以根据不同的场景进行选择使用。
1.DispatchSourceTimer: 基于GCD实现。
2.CADisplayLink:基于屏幕刷新实现。
3.Timer:基于RunLoop实现。

DispatchSourceTimer定时器
DispatchSourceTimer定时器:
可以通过DispatchSource.makeTimerSource(queue: DispatchQueue.main)方法来创建。
然后通过schedule(deadline: .now(), repeating: .seconds(1))方法指定定时器的初始延迟时间和重复时间间隔,
然后设置了一个事件处理程序来处理定时器要执行的操作。
最后调用timer.resume()启动定时器。
如果停止定时器,可以调用timer.cancel()方法。
优点为:Dispatch定时器非常轻量级,基于GCD的实现,可以利用GCD的优势来进行任务调度,性能高。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
var timer: DispatchSourceTimer?
func startCountdown() {
    //一般倒计时是操作UI,使用主队列
    timer = DispatchSource.makeTimerSource(queue: DispatchQueue.main)
    // //耗时操作放在全局队列,子线程处理
    // timer = DispatchSource.makeTimerSource(queue: DispatchQueue.global())
    timer.schedule(deadline: .now(), repeating: .seconds(1))
    timer.setEventHandler {
        // 定时器执行的操作
    }
    timer.resume()
}
 
deinit {
    timer.cancel()
    timer = nil
}
 
CADisplayLink定时器
CADisplayLink定时器:
可以通过CADisplayLink(target: self, selector: #selector(update))方法创建,
然后通过displayLink.add(to: .main, forMode: .common)方法将定时器添加到主运行循环中,并指定了运行模式,
然后定义一个update处理方法,该方法将在每个定时器周期中执行。
在对象销毁前停止定时器,可以调用displayLink.invalidate()方法。
它是和屏幕刷新率同步,优点在于精确度高,适用于需要频繁更新UI的场景。
CADisplayLink对象一旦创建就会运行,比较适合监控主线程UI卡顿。所以用做倒计时得场景下,更好的选择是使用DispatchSourceTimer定时器。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
var displayLink: CADisplayLink?
func startCountdown() {
    //一般倒计时是操作UI,使用主队列
    let displayLink = CADisplayLink(target: self, selector: #selector(update))
    //设置多长时间回调一次,默认每次刷新都会调用,大概60ps, 这里设置1表示1s调用一次
    displayLink.preferredFramesPerSecond = 1
    displayLink.add(to: .main, forMode: .common)
}
 
@objc func update() {
    // 定时器执行的操作
}
 
deinit {
    displayLink.invalidate()
    displayLink = nil
}
 
Timer定时器
Timer定时器:
可以使用Timer.scheduledTimer方法创建,然后指定重复间隔和一个闭包作为定时器要执行的操作。
然后将返回的定时器对象存储在成员变量timer中。
要停止定时器,可以调用timer.invalidate()方法。
Timer是一个简单的定时器,基于RunLoop的,通常用于实现对实时性要求不高的场合,因为它被注册在runloop的timers事件源集合中,如果当前runloop执行耗时任务超过了调用时间,那么就会丢弃当前次,直接执行下一次。导致定时器不准时的情况。
1
2
3
4
5
6
7
8
9
10
11
12
var timer: Timer?
func startCountdown() {
    //一般倒计时是操作UI,使用主队列
    timer = Timer.scheduledTimer(withTimeInterval: 1.0, repeats: true) { timer in
    // 定时器执行的操作
    }
}
 
deinit {
    timer.invalidate()
    timer = nil
}