android幾種定時器機制及區別

來源:互聯網
上載者:User

在android中,經常用到的定時器主要有以下幾種實現:
一、採用Handler與線程的sleep(long )方法
二、採用Handler的postDelayed(Runnable, long) 方法
三、採用Handler與timer及TimerTask結合的方法。
下面逐一介紹:
一、採用Handle與線程的sleep(long )方法
Handler主要用來處理接受到的訊息。這隻是最主要的方法,當然Handler裡還有其他的方法供實現,有興趣的可以去查API,這裡不過多解釋。
1.定義一個Handler類,用於處理接受到的Message.
Handler handler = new Handler() {
public void handleMessage(Message msg) {
//要做的事情
super.handleMessage(msg);
}
};
2.建立一個實現Runnable介面的線程類。如下:
public class MyThread implements Runnable{
@Override
public void run() {
// TODO Auto-generated method stub
while (true) {
try {
Thread.sleep(10000);//線程暫停10秒,單位毫秒
Message message=new Message();
message.what=1;
handler.sendMessage(message);//發送訊息
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
3.在需要啟動線程的地方加入下面語句:
new Thread(new MyThread()).start();
4.啟動線程後,線程每10s發送一次訊息。
二、採用Handler的postDelayed(Runnable, long) 方法
這個實現比較簡單一些:
1. Handler handler=new Handler();
Runnable runnable=new Runnable(){
@Override
public void run() {
// TODO Auto-generated method stub
//要做的事情
handler.postDelayed(this, 2000);
}
};
2.啟動計時器:
handler.postDelayed(runnable, 2000);//每兩秒執行一次runnable.
3.停止計時器:
handler.removeCallbacks(runnable);
三、採用Handler與timer及TimerTask結合的方法。
1.定義定時器、定時器任務及Handler控制代碼
private final Timer timer = new Timer();
private TimerTask task;
Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
// TODO Auto-generated method stub
//要做的事情
super.handleMessage(msg);
}

};
2.初始化計時器任務。
task = new TimerTask() {
@Override
public void run() {
// TODO Auto-generated method stub
Message message = new Message();
message.what = 1;
handler.sendMessage(message);
}
};
3.啟動定時器
timer.schedule(task, 2000, 2000);
簡要說一下上面三步提到的一些內容。
1.定時器任務(TimerTask)顧名思義,就是說當定時器到達指定的時間時要做的工作,這裡是想Handler發送一個訊息,由Handler類進行處理。
2. java.util.Timer.schedule(TimerTask task, long delay):這個方法是說,dalay/1000秒後執行task.只執行一次。
java.util.Timer.schedule(TimerTask task, long delay, long period):這個方法是說,delay/1000秒後執行task,然後進過period/1000秒再次執行task,這個用於迴圈任務,執行無數次,當然,你可以用timer.cancel();取消計時器的執行。

//////////////////////////////////////////////////////

然後是有關 Java.util.Timer 和 AlarmService的

在Android上常用的定時器有兩種,一種是Java.util.Timer,一種就是系統的AlarmService了。

實驗1:使用Java.util.Timer。
在onStart()創建立Timer,每5秒更新一次計數器,並啟動。
Java代碼
mTimer = new Timer();         
mTimer.schedule(new TimerTask() {             
            @Override 
            public void run() {  
                ++mCount;  
                mHandler.sendEmptyMessage(0);                 

            }  
        }, 5*1000, 5*1000);  

mTimer = new Timer();               
mTimer.schedule(new TimerTask() {                       
                        @Override
                        public void run() {
                                ++mCount;
                                mHandler.sendEmptyMessage(0);                               

                        }
                }, 5*1000, 5*1000);       

當串連USB線進行調試時,會發現一切工作正常,每5秒更新一次介面,即使是按下電源鍵,仍然會5秒觸發一次。
當拔掉USB線,按下電源鍵關閉螢幕後,過一段時間再開啟,發現定時器明顯沒有繼續計數,停留在了關閉電源鍵時的數字。

實驗2:使用AlarmService:
2.1通過AlarmService每個5秒發送一個廣播,setRepeating時的類型為AlarmManager.ELAPSED_REALTIME。
Java代碼
AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE);  
am.setRepeating(AlarmManager.ELAPSED_REALTIME, firstTime, 5*1000, sender); 

AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE);
am.setRepeating(AlarmManager.ELAPSED_REALTIME, firstTime, 5*1000, sender);

拔掉USB線,按下電源鍵,過一段時間再次開啟螢幕,發現定時器沒有繼續計數。
2.2setRepeating是的類型設定為AlarmManager.ELAPSED_REALTIME_WAKEUP
Java代碼
AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE);  
am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, firstTime, 5*1000, sender); 

AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE);
am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, firstTime, 5*1000, sender);

拔掉USB線,按下電源鍵,過一點時間再次開啟螢幕,發現定時器一直在計數。

如此看來,使用WAKEUP才能保證自己想要的定時器一直工作,但是肯定會引起耗電量的增加。

///////////////////////////////////////

最後是有關schedule和scheduleAtFixedRate區別

schedule和scheduleAtFixedRate 區別:

(1) 2個參數的schedule在制定任務計劃時,如果指定的計劃執行時間scheduledExecutionTime<= systemCurrentTime,則task會被立即執行。scheduledExecutionTime不會因為某一個task的過度執行而改變。
(2) 3個參數的schedule在制定反覆執行一個task的計劃時,每一次執行這個task的計劃執行時間隨著前一次的實際執行時間而變,也就是 scheduledExecutionTime(第n+1次)=realExecutionTime(第n次)+periodTime。也就是說如果第n 次執行task時,由於某種原因這次執行時間過長,執行完後的systemCurrentTime>= scheduledExecutionTime(第n+1次),則此時不做時隔等待,立即執行第n+1次task,而接下來的第n+2次task的
scheduledExecutionTime(第n+2次)就隨著變成了realExecutionTime(第n+1次)+periodTime。說白了,這個方法更注重保持間隔時間的穩定。
(3)3個參數的scheduleAtFixedRate在制定反覆執行一個task的計劃時,每一次執行這個task的計劃執行時間在最初就被定下來了,也就是scheduledExecutionTime(第n次)=firstExecuteTime +n*periodTime;如果第n次執行task時,由於某種原因這次執行時間過長,執行完後的systemCurrentTime>= scheduledExecutionTime(第n+1次),則此時不做period間隔等待,立即執行第n+1次task,而接下來的第n+2次的
task的scheduledExecutionTime(第n+2次)依然還是firstExecuteTime+(n+2)*periodTime這在第一次執行task就定下來了。說白了,這個方法更注重保持執行頻率的穩定。

Timer的執行個體:

package com.hemes.timer;

import java.util.*;

public class doTask extends TimerTask {

    // true時使用後台進程線程。只要剩下的程式記敘運行,後台進程線程就會執行。
    Timer myTimer;

    public void start(int delay, int hour) {
    myTimer = new Timer();
        myTimer.schedule(this, delay * 1000, hour*1000*60*60); //利用timer.schedule方法
      
        //public void schedule(TimerTask task,long time,long period)
        //task被安排在延遲time後執行,執行後將每隔period(毫秒)反覆執行。由於規定的時間間隔並不能保證與時鐘精準的同不步,所以該方
    }
  
    public void start(Date time, int hour) {
    myTimer = new Timer();
        myTimer.schedule(this, time, hour*1000*60*60); //利用timer.schedule方法
      
        //public void schedule(TimerTask task,Date time,long period)
        //task被安排在time指定的時間執行,執行後將每隔period(毫秒)反覆執行。由於規定的時間間隔並不能保證與時鐘精準的同不步,所以該方
    }
  
    public void run() {
    //執行任務(sql)
        System.out.println("do Task...");
    }
  
    public void end(){
        myTimer.cancel();
        //終止Timer的功能執行,但不會對正在執行的任務有影響。當執行cancel方法後將不能再用其分配任務。
    }
  
    //測試
    public static void main(String args[]) {
  
        doTask myTask1 = new doTask();
      
        //Get the Date corresponding to 11:30:00 pm today.
        Calendar calendar = Calendar.getInstance();
        calendar.set(Calendar.HOUR_OF_DAY, 23);
        calendar.set(Calendar.MINUTE, 30);
        calendar.set(Calendar.SECOND, 0);
        Date time = calendar.getTime();

        myTask1.start(time,24);
      
      
     
        //myTask1.end();//線程結束
     
    }
}

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.