title: Android CountDownTimer源碼解析
date: 2016-04-02
tags: CountDownTimer
CountDownTimer是android sdk中os包下的一個輔助抽象類,這個類通過handler來實現(xiàn)一個倒計時的操作。在倒計時期間會定期調(diào)用用戶實現(xiàn)的回調(diào)函數(shù)信姓。
比如一個簡單的使用場景:一個30秒倒計時
new CountDownTimer(30000, 1000) {
public void onTick(long millisUntilFinished) {
mTextField.setText("seconds remaining: " + millisUntilFinished / 1000);
}
public void onFinish() {
mTextField.setText("done!");
}
}.start();
onTick方法對這個對象來說是同步的,當onTick方法的實現(xiàn)花費的時間和倒計時的間隔來說需要很久的時候粉臊,調(diào)用onTick也是按照順序發(fā)生的,方法不會在之前的回調(diào)完成之前發(fā)生驶兜。
private long mStopTimeInFuture;
/**
* 是否取消
*/
private boolean mCancelled = false;
CountDownTime的構造函數(shù)扼仲,millisInFuture從調(diào)用start方法開始直到倒計時結束的毫秒時間,countDownInterval 接收onTick回調(diào)的時間間隔
public CountDownTimer(long millisInFuture, long countDownInterval) {
mMillisInFuture = millisInFuture;
mCountdownInterval = countDownInterval;
}
取消倒計時
private static final int MSG = 1;
public synchronized final void cancel() {
mCancelled = true;
mHandler.removeMessages(MSG);
}
開始倒計時
public synchronized final CountDownTimer start() {
mCancelled = false;
if (mMillisInFuture <= 0) {
onFinish();
return this;
}
mStopTimeInFuture = SystemClock.elapsedRealtime() + mMillisInFuture;
mHandler.sendMessage(mHandler.obtainMessage(MSG));
return this;
}
定期執(zhí)行的回調(diào)抄淑,使用的時候需要實現(xiàn)這個方法屠凶,millisUntilFinished參數(shù)表示剩余時間
public abstract void onTick(long millisUntilFinished);
當?shù)褂嫊r結束的時候的回調(diào)
public abstract void onFinish();
處理倒計時的handler
private Handler mHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
synchronized (CountDownTimer.this) {
if (mCancelled) {
return;
}
final long millisLeft = mStopTimeInFuture - SystemClock.elapsedRealtime();
if (millisLeft <= 0) {
onFinish();
} else if (millisLeft < mCountdownInterval) {
// 剩余時間小于一次時間間隔的時候,不再通知肆资,只是延遲一下
sendMessageDelayed(obtainMessage(MSG), millisLeft);
} else {
long lastTickStart = SystemClock.elapsedRealtime();
onTick(millisLeft);
// 處理用戶onTick執(zhí)行的時間
long delay = lastTickStart + mCountdownInterval - SystemClock.elapsedRealtime();
// 特殊情況:用戶的onTick方法花費的時間比interval長矗愧,那么直接跳轉(zhuǎn)到下一次interval
while (delay < 0) delay += mCountdownInterval;
sendMessageDelayed(obtainMessage(MSG), delay);
}
}
}
};
通過源碼可知,CountDownTimer采用的是handler機制郑原,通過sendMessageDelayed延遲發(fā)送一條message到主線程的looper中唉韭,然后在自身中收到之后判斷剩余時間夜涕,并發(fā)出相關回調(diào),然后再次發(fā)出message的方式属愤。之前實現(xiàn)這種倒計時是通過asynctask女器,在線程中通過Thread.sleep來實現(xiàn),通過asyntask的cancel來實現(xiàn)取消住诸,通過構造asynctask傳入接口的實現(xiàn)來onTick的類似功能驾胆。這個CountDownTimer默認是在當前l(fā)ooper當中,可以是在UI線程也可以是在非UI線程中執(zhí)行贱呐,如果在UI線程中執(zhí)行丧诺,那是不是會稍微加重UI線程的負擔?