在Android上經常使用的定時器有兩種,一種是Java.util.Timer,一種就是系統的AlarmServicejava
AlarmManager的做用文檔中的解釋是:在特定的時刻爲咱們廣播一個指定的Intent。簡單的說就是咱們設定一個時間,而後在該時間到來時,AlarmManager爲咱們廣播一個咱們設定的Intent,經常使用方法有五個: (1)set(int type,long startTime,PendingIntent pi); 該方法用於設置一次性鬧鐘,第一個參數表示鬧鐘類型,第二個參數表示鬧鐘執行時間,第三個參數表示鬧鐘響應動做。 (2)setRepeating(int type,long startTime,long intervalTime,PendingIntent pi);android
該方法用於設置重複鬧鐘,第一個參數表示鬧鐘類型,第二個參數表示鬧鐘首次執行時間,第三個參數表示鬧鐘兩次執行的間隔時間,第四個參數表示鬧鐘響應動做。相似JAVA的Timer裏面scheduleAtFixedRate(TimerTask task, long delay, long period):在反覆執行一個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就定下來了。說白了,這個方法更注重保持執行頻率的穩定。
(3)setInexactRepeating(int type,long startTime,long intervalTime,PendingIntent pi); 該方法也用於設置重複鬧鐘,與第二個方法類似,不過其兩個鬧鐘執行的間隔時間不是固定的而已。它相對而言更節能(power-efficient)一些,由於系統可能會將幾個差很少的鬧鐘合併爲一個來執行,減小設備的喚醒次數。 有點相似JAVA的Timer裏面schedule(TimerTask task, Date firstTime, long period):在反覆執行一個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。說 白了,這個方法更注重保持間隔時間的穩定。this
(4)cancel(PendingIntent operation) 取消一個設置的鬧鐘code
(5)setTimeZone(String timeZone) 設置系統的默認時區。須要android.permission.SET_TIME_ZONE權限xml
三個方法各個參數: (1)int type:鬧鐘的類型,經常使用的有5個值:對象
[java] view plaincopyprint?AlarmManager.ELAPSED_REALTIME
AlarmManager.ELAPSED_REALTIME_WAKEUP
AlarmManager.RTC
AlarmManager.RTC_WAKEUP
AlarmManager.POWER_OFF_WAKEUPci
AlarmManager.ELAPSED_REALTIME AlarmManager.ELAPSED_REALTIME_WAKEUP AlarmManager.RTC AlarmManager.RTC_WAKEUP AlarmManager.POWER_OFF_WAKEUP AlarmManager.ELAPSED_REALTIME當系統進入睡眠狀態時,這種類型的鬧鈴不會喚醒系統。直到系統下次被喚醒才傳遞它,該鬧鈴所用的時間是相對時間,是從系統啓動後開始計時的,包括睡眠時間,能夠經過調用SystemClock.elapsedRealtime()得到。系統值是3 (0x00000003)文檔
AlarmManager.ELAPSED_REALTIME_WAKEUP表示鬧鐘在睡眠狀態下會喚醒系統並執行提示功能,該狀態下鬧鐘也使用相對時間,用法同ELAPSED_REALTIME,系統值是2 (0x00000002) AlarmManager.RTC表示鬧鐘在睡眠狀態下,這種類型的鬧鈴不會喚醒系統。直到系統下次被喚醒才傳遞它,該鬧鈴所用的時間是絕對時間,所用時間是UTC時間,能夠經過調用 System.currentTimeMillis()得到。系統值是1 (0x00000001) AlarmManager.RTC_WAKEUP表示鬧鐘在睡眠狀態下會喚醒系統並執行提示功能,該狀態下鬧鐘使用絕對時間,系統值爲0(0x00000000); AlarmManager.POWER_OFF_WAKEUP表示鬧鐘在手機關機狀態下也能正常進行提示功能(關機鬧鐘),因此是5個狀態中用的最多的狀態之一,該狀態下鬧鐘也是用絕對時間,系統值爲4(0x00000004);不過本狀態好像受SDK版本影響,某些版本並不支持;
(2)long startTime:字符串
鬧鐘的第一次執行時間,以毫秒爲單位,能夠自定義時間,不過通常使用當前時間。須要注意的是,本屬性與第一個屬性(type)密切相關, 若是第一個參數對應的鬧鐘使用的是相對時間(ELAPSED_REALTIME和ELAPSED_REALTIME_WAKEUP),那麼本屬性就得使用相對時間(相對於系統啓動時間來講),好比當前時間就表示爲:SystemClock.elapsedRealtime(); 若是第一個參數對應的鬧鐘使用的是絕對時間(RTC、RTC_WAKEUP、POWER_OFF_WAKEUP),那麼本屬性就得使用絕對時間,好比當前時間就表示爲:System.currentTimeMillis()。
(3)long intervalTime:get
對於後兩個方法來講,存在本屬性,表示兩次鬧鐘執行的間隔時間,也是以毫秒爲單位。
(4)PendingIntent pi:
是鬧鐘的執行動做,好比發送一個廣播、給出提示等等。PendingIntent是Intent的封裝類。須要注意的是,若是是經過啓動服務來實現鬧鐘提示的話,PendingIntent對象的獲取就應該採用Pending.getService(Context c,int i,Intent intent,int j)方法;若是是經過廣播來實現鬧鐘提示的話,PendingIntent對象的獲取就應該採用PendingIntent.getBroadcast(Context c,int i,Intent intent,int j)方法;若是是採用Activity的方式來實現鬧鐘提示的話,PendingIntent對象的獲取就應該採用PendingIntent.getActivity(Context c,int i,Intent intent,int j)方法。若是這三種方法錯用了的話,雖然不會報錯,可是看不到鬧鐘提示效果。
AlarmManager使用示例:利用用戶自定義廣播實現鬧鐘功能,從當前時間開始,每隔10分鐘提示一次
(1)實現原理:在SendActivity.java中定義一個AlarmManager對象,指定該對象從當前時間開始,每隔10分鐘向名爲「MYALARMRECEIVER」的廣播接收器發出一條廣播,附加消息內容爲「你該打醬油了」;建立一個名爲MyReceiver的廣播接收器,在其onReceive方法中獲取Intent對象傳過來的值(「你該打醬油了」)並用一個Toast組件顯示出來;在AndroidManifest.xml文件中註冊SendActivity類和廣播接收器類MyReceiver,設置MyReceiver的action的值爲「MYALARMRECEIVER」
(2)代碼實現:
建立廣播接收類MyReceiver.java,在其onReceive方法中獲取Intent的附加信息msg,並用Toast組件顯示
[java] view plaincopyprint?public void onReceive(Context context,Intent intent){
String msg = intent.getStringExtra("msg");
Toast.makeText(context,msg,Toast.LENGTH_SHORT).show();
}
public void onReceive(Context context,Intent intent){ String msg = intent.getStringExtra("msg"); Toast.makeText(context,msg,Toast.LENGTH_SHORT).show(); } 建立SendActivity.java,用於設置鬧鐘,定時發出廣播
[java] view plaincopyprint?//建立Intent對象,action指向廣播接收類,附加信息爲字符串「你該打醬油了」
Intent intent = new Intent("MYALARMRECEIVER");
intent.putExtra("msg","你該打醬油了");
//建立PendingIntent對象封裝Intent,因爲是使用廣播,注意使用getBroadcast方法
PendingIntent pi = PendingIntent.getBroadcast(this,0,intent,0);
//獲取AlarmManager對象
AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE);
//設置鬧鐘從當前時間開始,每隔10分鐘執行一次PendingIntent對象,注意第一個參數與第二個參數的關係
am.setRepeating(AlarmManager.RTC_WAKEUP,System.currentMillis(),600*1000,pi);
//建立Intent對象,action指向廣播接收類,附加信息爲字符串「你該打醬油了」
Intent intent = new Intent("MYALARMRECEIVER");
intent.putExtra("msg","你該打醬油了");
//建立PendingIntent對象封裝Intent,因爲是使用廣播,注意使用getBroadcast方法
PendingIntent pi = PendingIntent.getBroadcast(this,0,intent,0);
//獲取AlarmManager對象
AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE);
//設置鬧鐘從當前時間開始,每隔10分鐘執行一次PendingIntent對象,注意第一個參數與第二個參數的關係
am.setRepeating(AlarmManager.RTC_WAKEUP,System.currentMillis(),600*1000,pi); 有時候,也許咱們須要同時開啓多個定時器,咱們先來看看下面這段代碼:
[java] view plaincopyprint?AlarmManager am = null;
am = (AlarmManager) context.getApplicationContext().getSystemService(Context.ALARM_SERVICE);
for (int i = 0; i < 10; i ++) {
...
Intent i = new Intent("xxx");
PendingIntent sender = PendingIntent.getBroadcast(context.getApplicationContext(), 0, i, PendingIntent.FLAG_UPDATE_CURRENT);
...
am.setRepeating(...);
}
AlarmManager am = null; am = (AlarmManager) context.getApplicationContext().getSystemService(Context.ALARM_SERVICE); for (int i = 0; i < 10; i ++) { ... Intent i = new Intent("xxx"); PendingIntent sender = PendingIntent.getBroadcast(context.getApplicationContext(), 0, i, PendingIntent.FLAG_UPDATE_CURRENT); ... am.setRepeating(...); } 若是採用這種作法後面的定時器會將前面的定時器"覆蓋"掉,只會啓動最後一個定時器
解決辦法
PendingIntent.getBroadcast(Context context, int requestCode, Intent intent, int flags);
第二個參數requestCode必定要是惟一的,好比不一樣的ID之類的,(若是系統須要多個定時器的話)。