Timer → Coroutines
老写法(Java)
java
// 延时执行一次
new Timer().schedule(new TimerTask() {
@Override
public void run() {
doWork();
}
}, 3000);
// 定时重复执行
Timer timer = new Timer();
timer.scheduleAtFixedRate(new TimerTask() {
@Override
public void run() {
fetchData();
}
}, 0, 5000);
// 结束时取消
timer.cancel();
问题在哪里
Timer 用单个后台线程执行所有任务,一个 TimerTask 抛异常整个 Timer 直接挂掉,其他排队的任务全部作废。而且 Timer 不感知 Android 生命周期,忘记 cancel 会导致后台任务持续运行消耗电量。
新写法(Kotlin + Coroutines)
kotlin
// 延时执行一次
viewModelScope.launch {
delay(3000)
doWork()
}
// 定时重复执行
viewModelScope.launch {
while (isActive) {
fetchData()
delay(5000)
}
}
// 需要精确固定间隔(类似 scheduleAtFixedRate)
viewModelScope.launch {
var nextTime = System.currentTimeMillis()
while (isActive) {
fetchData()
nextTime += 5000
val delayTime = nextTime - System.currentTimeMillis()
if (delayTime > 0) delay(delayTime)
}
}
一句话注意
while (isActive) 是关键------isActive 是协程的属性,协程被取消或 ViewModel 清除后变为 false,自动跳出循环。不限时长运行的循环必须检查这个条件,否则协程取消不了。
Timer 的 schedule vs scheduleAtFixedRate 的区别:schedule 等上一个任务执行完才开始计时间隔,scheduleAtFixedRate 按固定周期执行不管上一个任务是否完成。协程中默认 delay 行为类似 schedule。如果需要固定频率,用上面第三个例子记录时间戳补偿每次的延迟。
Java Android 老项目迁移系列,持续更新中。