Android AlarmManager設定定時事件提醒
最近用到了AlarmManager,遇到了問題,當我設定多個定時器時,發現只有一個起作用,百思不得其解,後來在網上找到了原因,把人家的解決辦法貼上:
AlarmManager的常用方法有三個:
(1)set(int type,long startTime,PendingIntent pi);
該方法用於設定一次性鬧鐘,第一個引數表示鬧鐘型別,第二個引數表示鬧鐘執行時間,第三個引數表示鬧鐘響應動作。
(2)setRepeating(int type,long startTime,long intervalTime,PendingIntent pi);
(3)setInexactRepeating(int type,long startTime,long intervalTime,PendingIntent pi);
該方法也用於設定重複鬧鐘,與第二個方法相似,不過其兩個鬧鐘執行的間隔時間不是固定的而已。
三個方法各個引數:
(1)int type:鬧鐘的型別,常用的有5個值:
AlarmManager.ELAPSED_REALTIME AlarmManager.ELAPSED_REALTIME_WAKEUP AlarmManager.RTC AlarmManager.RTC_WAKEUP AlarmManager.POWER_OFF_WAKEUP
AlarmManager.ELAPSED_REALTIME表示鬧鐘在手機睡眠狀態下不可用,該狀態下鬧鐘使用相對時間(相對於系統啟動開始),狀態值為3;
AlarmManager.ELAPSED_REALTIME_WAKEUP表示鬧鐘在睡眠狀態下會喚醒系統並執行提示功能,該狀態下鬧鐘也使用相對時間,狀態值為2;
AlarmManager.RTC表示鬧鐘在睡眠狀態下不可用,該狀態下鬧鐘使用絕對時間,即當前系統時間,狀態值為1;
AlarmManager.RTC_WAKEUP表示鬧鐘在睡眠狀態下會喚醒系統並執行提示功能,該狀態下鬧鐘使用絕對時間,狀態值為0;
AlarmManager.POWER_OFF_WAKEUP表示鬧鐘在手機關機狀態下也能正常進行提示功能,所以是5個狀態中用的最多的狀態之一,該狀態下鬧鐘也是用絕對時間,狀態值為4;不過本狀態好像受SDK版本影響,某些版本並不支援;
(2)long startTime:
鬧鐘的第一次執行時間,以毫秒為單位,可以自定義時間,不過一般使用當前時間。需要注意的是,本屬性與第一個屬性(type)密切相關,
如果第一個引數對應的鬧鐘使用的是相對時間(ELAPSED_REALTIME和ELAPSED_REALTIME_WAKEUP),那麼本屬性就得使用相對時間(相對於系統啟動時間來說),比如當前時間就表示為:SystemClock.elapsedRealtime();
如果第一個引數對應的鬧鐘使用的是絕對時間(RTC、RTC_WAKEUP、POWER_OFF_WAKEUP),那麼本屬性就得使用絕對時間,比如當前時間就表示為:System.currentTimeMillis()。
(3)long intervalTime:
對於後兩個方法來說,存在本屬性,表示兩次鬧鐘執行的間隔時間,也是以毫秒為單位。
(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)方法。如果這三種方法錯用了的話,雖然不會報錯,但是看不到鬧鐘提示效果。
Android定時器AlarmManager就說這麼多
參見下面的程式碼:
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之類的,(如果系統需要多個定時器的話)。
簡單實現的程式碼如下:
package com.example.alarmmanagerdemo;
import java.util.Calendar;
import java.util.TimeZone;
import android.app.Activity;
import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.Intent;
import android.os.Bundle;
import android.os.SystemClock;
import android.util.Log;
import android.view.Menu;
import android.view.View;
import android.widget.Button;
import android.widget.TimePicker;
import android.widget.Toast;
/**
*
* @ClassName: MainActivity
* @Description: 主介面
*
*/
public class MainActivity extends Activity {
private static final String TAG = MainActivity.class.getSimpleName();
private TimePicker mTimePicker;
private Button mButton1;
private Button mButton2;
private Button mButtonCancel;
private int mHour = -1;
private int mMinute = -1;
public static final long DAY = 1000L * 60 * 60 * 24;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// 獲取當前時間
Calendar calendar = Calendar.getInstance();
calendar.setTimeZone(TimeZone.getTimeZone("GMT+8"));
if(mHour == -1 && mMinute == -1) {
mHour = calendar.get(Calendar.HOUR_OF_DAY);
mMinute = calendar.get(Calendar.MINUTE);
}
mTimePicker = (TimePicker)findViewById(R.id.timePicker);
mTimePicker.setCurrentHour(mHour);
mTimePicker.setCurrentMinute(mMinute);
mTimePicker.setOnTimeChangedListener(new TimePicker.OnTimeChangedListener() {
@Override
public void onTimeChanged(TimePicker view, int hourOfDay, int minute) {
mHour = hourOfDay;
mMinute = minute;
}
});
mButton1 = (Button)findViewById(R.id.normal_button);
mButton1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(MainActivity.this, AlarmReceiver.class);
PendingIntent sender = PendingIntent.getBroadcast(MainActivity.this, 0, intent, 0);
// 過20s 執行這個鬧鈴
Calendar calendar = Calendar.getInstance();
calendar.setTimeInMillis(System.currentTimeMillis());
calendar.setTimeZone(TimeZone.getTimeZone("GMT+8"));
calendar.add(Calendar.SECOND, 20);
// 進行鬧鈴註冊
AlarmManager manager = (AlarmManager)getSystemService(ALARM_SERVICE);
manager.set(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), sender);
Toast.makeText(MainActivity.this, "設定簡單鬧鈴成功!", Toast.LENGTH_LONG).show();
}
});
mButton2 = (Button)findViewById(R.id.repeating_button);
mButton2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(MainActivity.this, AlarmReceiver.class);
PendingIntent sender = PendingIntent.getBroadcast(MainActivity.this, 0, intent, 0);
long firstTime = SystemClock.elapsedRealtime(); // 開機之後到現在的執行時間(包括睡眠時間)
long systemTime = System.currentTimeMillis();
Calendar calendar = Calendar.getInstance();
calendar.setTimeInMillis(System.currentTimeMillis());
calendar.setTimeZone(TimeZone.getTimeZone("GMT+8")); // 這裡時區需要設定一下,不然會有8個小時的時間差
calendar.set(Calendar.MINUTE, mMinute);
calendar.set(Calendar.HOUR_OF_DAY, mHour);
calendar.set(Calendar.SECOND, 0);
calendar.set(Calendar.MILLISECOND, 0);
// 選擇的每天定時時間
long selectTime = calendar.getTimeInMillis();
// 如果當前時間大於設定的時間,那麼就從第二天的設定時間開始
if(systemTime > selectTime) {
Toast.makeText(MainActivity.this, "設定的時間小於當前時間", Toast.LENGTH_SHORT).show();
calendar.add(Calendar.DAY_OF_MONTH, 1);
selectTime = calendar.getTimeInMillis();
}
// 計算現在時間到設定時間的時間差
long time = selectTime - systemTime;
firstTime += time;
// 進行鬧鈴註冊
AlarmManager manager = (AlarmManager)getSystemService(ALARM_SERVICE);
manager.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP,
firstTime, 10*1000, sender);
Log.i(TAG, "time ==== " + time + ", selectTime ===== "
+ selectTime + ", systemTime ==== " + systemTime + ", firstTime === " + firstTime);
Toast.makeText(MainActivity.this, "設定重複鬧鈴成功! ", Toast.LENGTH_LONG).show();
}
});
mButtonCancel = (Button)findViewById(R.id.cancel_button);
mButtonCancel.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(MainActivity.this, AlarmReceiver.class);
PendingIntent sender = PendingIntent.getBroadcast(MainActivity.this,
0, intent, 0);
// 取消鬧鈴
AlarmManager am = (AlarmManager)getSystemService(ALARM_SERVICE);
am.cancel(sender);
}
});
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
}
AlarmReceiver <span style="font-family: Arial; background-color: rgb(255, 255, 255);">如下:</span>
/**
*
* @ClassName: AlarmReceiver
* @Description: 鬧鈴時間到了會進入這個廣播,這個時候可以做一些該做的業務。
* @author zk
*/
public class AlarmReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
Toast.makeText(context, "鬧鈴響了", Toast.LENGTH_LONG).show();
// NotificationManager nm=(NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
// Notification notification=new Notification(R.drawable.ic_launcher,"時間到了",1000);
// notification.flags=Notification.DEFAULT_ALL;
//// notification.defaults=Notification.DEFAULT_ALL;
// nm.notify(0, notification);
}
}
AlarmReceiver 註冊<span style="font-family: Arial; background-color: rgb(255, 255, 255);">檔案:</span>
<receiver android:name="com.example.alarmmanagerdemo.AlarmReceiver" android:process=":remote">