AlarmManager实质是一个全局的定时器,是Android中经常使用的一种系统级别的提示服务,在指定时间或周期性启动其它组件(包括Activity,Service,BroadcastReceiver)。本文将讲解一下如何使用AlarmManager实现定时提醒功能。
java
Intent intent = new Intent();
intent.setAction(GlobalValues.TIMER_ACTION_REPEATING);
PendingIntent sender = PendingIntent.getBroadcast(context, 0, intent, 0);
AlarmManager alarm = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
alarm.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis() + 5 * 1000, 3 * 1000, sender);复制代码
setRepeating(int type,long startTime,long intervalTime,PendingIntent pi)复制代码
该方法用于设置周期性执行的定时服务。type:闹钟类型,startTime:闹钟首次执行时间,intervalTime:闹钟两次执行的间隔时间,pi:闹钟响应动做。 android
setInexactRepeating(int type,long startTime,long intervalTime,PendingIntent pi)复制代码
该方法也用于设置周期定式服务,与上一种相似。不过其两个闹钟执行的间隔时间不是固定的。它相对而言更省电一些,由于系统可能会将几个差很少的闹钟合并为一个来执行,减小设备的唤醒次数。 app
间隔一天: INTERVAL_DAY
间隔半天: INTERVAL_HALF_DAY
间隔15分钟: INTERVAL_FIFTEEN_MINUTES
间隔半个小时: INTERVAL_HALF_HOUR
间隔一个小时: INTERVAL_HOUR ide
//得到系统提供的AlarmManager服务的对象
AlarmManager alarm = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
//Intent设置要启动的组件,这里启动广播
Intent myIntent = new Intent();
myIntent.setAction(GlobalValues.TIMER_ACTION);
//PendingIntent对象设置动做,启动的是Activity仍是Service,或广播!
PendingIntent sender = PendingIntent.getBroadcast(context, 0, myIntent,0);
//注册闹钟
alarm.set(AlarmManager.RTC_WAKEUP, System.currentTimeMillis() + 5 * 1000, sender);复制代码
set(int type,long startTime,PendingIntent pi)复制代码
该方法用于设置一次性定时服务。type:闹钟类型,startTime:闹钟执行时间,pi:闹钟响应动做。 工具
Intent myIntent = new Intent();
myIntent.setAction(GlobalValues.TIMER_ACTION);
//myIntent.setAction(GlobalValues.TIMER_ACTION_REPEATING);
PendingIntent sender = PendingIntent.getBroadcast(context, 0, myIntent,0);
AlarmManager alarm = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
alarm.cancel(sender);复制代码
若连续设置多个闹钟,则只有最后一个闹钟会生效,那么这种状况咱们怎么处理呢?其实很简单。咱们能够给每一个闹钟设置惟一的id,传入getBroadcast()第二个参数。在这里我是每设置一个id则自增1存入Shareprefrence里,保证id惟一性。ui
//给每一个闹钟设置不一样ID防止覆盖
int alarmId = SharedPreUtils.getInteger(context, "alarm_id", 0);
SharedPreUtils.setInteger(context, "alarm_id", ++alarmId);
PendingIntent sender = PendingIntent.getBroadcast(context, alarmId, myIntent, 0);复制代码
在取消闹钟时咱们也能够根据这个id关闭不一样的闹钟。 spa
ELAPSED_REALTIME:
在指定的延时事后,发送广播,但不唤醒设备(闹钟在睡眠状态下不可用)。若是在系统休眠时闹钟触发,它将不会被传递,直到下一次设备唤醒。
ELAPSED_REALTIME_WAKEUP:
在指定的延时事后,发送广播,并唤醒设备(即便关机也会执行operation所对应的组件) 。延时是要把系统启动的时间SystemClock.elapsedRealtime()
算进去的。
RTC:
指定当系统调用System.currentTimeMillis()
方法返回的值与triggerAtTime相等时启动operation所对应的设备(在指定的时刻,发送广播,但不唤醒设备)。若是在系统休眠时闹钟触发,它将不会被传递,直到下一次设备唤醒(闹钟在睡眠状态下不可用)。
RTC_WAKEUP:
指定当系统调用System.currentTimeMillis()
方法返回的值与triggerAtTime相等时启动operation所对应的设备(在指定的时刻,发送广播,并唤醒设备)。即便系统关机也会执行operation所对应的组件。
POWER_OFF_WAKEUP:
表示闹钟在手机关机状态下也能正常进行提示功能,因此是5个状态中用的最多的状态之一,该状态下闹钟也是用绝对时间,状态值为4;不过本状态好像受SDK版本影响,某些版本并不支持。 code
闹钟的第一次执行时间,以毫秒为单位,能够自定义时间,不过通常使用当前时间。须要注意的是,本属性与第一个属性(type)密切相关,若是第一个参数对应的闹钟使用的是相对时间(ELAPSED_REALTIME和ELAPSED_REALTIME_WAKEUP),那么本属性就得使用相对时间(相对于系统启动时间来讲),好比当前时间就表示为:SystemClock.elapsedRealtime();
若是第一个参数对应的闹钟使用的是绝对时间(RTC、RTC_WAKEUP、POWER_OFF_WAKEUP),那么本属性就得使用绝对时间,好比当前时间就表示为:System.currentTimeMillis()
xml
对于周期定时方式来讲,存在本属性,表示两次闹钟执行的间隔时间,也是以毫秒为单位。对象
是闹钟的执行动做,好比发送一个广播、给出提示等等。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)
方法。若是这三种方法错用了的话,虽然不会报错,可是看不到闹钟提示效果。。
public class AlarmReceiver extends BroadcastReceiver {
private NotificationManager m_notificationMgr = null;
private static final int NOTIFICATION_FLAG = 3;
@Override
public void onReceive(Context context, Intent intent) {
m_notificationMgr = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVIC if (intent.getAction().equals(GlobalValues.TIMER_ACTION_REPEATING)) {
Log.e("alarm_receiver", "周期闹钟");
} else if (intent.getAction().equals(GlobalValues.TIMER_ACTION)) {
Log.e("alarm_receiver", "定时闹钟");
Bitmap bitmap = BitmapFactory.decodeResource(context.getResources(), R.drawable.logo);
Intent intent1 = new Intent(context, WriteDiaryActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(context, 0, intent1, 0);
Notification notify = new Notification.Builder(context)
.setSmallIcon(R.drawable.logo) // 设置状态栏中的小图片,尺寸通常建议在24×24
.setLargeIcon(bitmap) // 这里也能够设置大图标
.setTicker("亲情日历") // 设置显示的提示文字
.setContentTitle("亲情日历") // 设置显示的标题
.setContentText("您有日记提醒哦") // 消息的详细内容
.setContentIntent(pendingIntent) // 关联PendingIntent
.setNumber(1) // 在TextView的右方显示的数字,能够在外部定义一个变量,点击累加setNumber(count),这时显示的和
.getNotification(); // 须要注意build()是在API level16及以后增长的,在API11中可使用getNotificatin()来
notify.flags |= Notification.FLAG_AUTO_CANCEL;
NotificationManager manager = (NotificationManager) context.getSystemService(Context.NOTIF
manager.notify(NOTIFICATION_FLAG, notify);
bitmap.recycle(); //回收bitmap
}
}
}复制代码
最后别忘了在清单里注册广播。
<!--闹钟接收广播-->
<receiver android:name=".util.service.AlarmReceiver">
<intent-filter>
<action android:name="com.e_eduspace.TIMER_ACTION_REPEATING" />
<action android:name="com.e_eduspace.TIMER_ACTION" />
</intent-filter>
</receiver>复制代码
public class GlobalValues {
// 周期性的闹钟
public final static String TIMER_ACTION_REPEATING = "com.e_eduspace.TIMER_ACTION_REPEATING";
// 定时闹钟
public final static String TIMER_ACTION = "com.e_eduspace.TIMER_ACTION";
}复制代码
package com.e_eduspace.familycalendar.util;
import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import com.prolificinteractive.materialcalendarview.CalendarDay;
/** * 闹钟定时工具类 * * @author xulei * @time 2016/12/13 10:03 */
public class AlarmTimer {
/** * 设置周期性闹钟 * * @param context * @param firstTime * @param cycTime * @param action * @param AlarmManagerType 闹钟的类型,经常使用的有5个值:AlarmManager.ELAPSED_REALTIME、 * AlarmManager.ELAPSED_REALTIME_WAKEUP、AlarmManager.RTC、 * AlarmManager.RTC_WAKEUP、AlarmManager.POWER_OFF_WAKEUP */
public static void setRepeatingAlarmTimer(Context context, long firstTime, long cycTime, String action, int AlarmManagerType) {
Intent myIntent = new Intent();
myIntent.setAction(action);
PendingIntent sender = PendingIntent.getBroadcast(context, 0, myIntent, 0);
AlarmManager alarm = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
alarm.setRepeating(AlarmManagerType, firstTime, cycTime, sender);
//param1:闹钟类型,param1:闹钟首次执行时间,param1:闹钟两次执行的间隔时间,param1:闹钟响应动做。
}
/** * 设置定时闹钟 * * @param context * @param cycTime * @param action * @param AlarmManagerType 闹钟的类型,经常使用的有5个值:AlarmManager.ELAPSED_REALTIME、 * AlarmManager.ELAPSED_REALTIME_WAKEUP、AlarmManager.RTC、 * AlarmManager.RTC_WAKEUP、AlarmManager.POWER_OFF_WAKEUP */
public static void setAlarmTimer(Context context, long cycTime, String action, int AlarmManagerType, CalendarDay date) {
Intent myIntent = new Intent();
//传递定时日期
myIntent.putExtra("date", date);
myIntent.setAction(action);
//给每一个闹钟设置不一样ID防止覆盖
int alarmId = SharedPreUtils.getInteger(context, "alarm_id", 0);
SharedPreUtils.setInteger(context, "alarm_id", ++alarmId);
PendingIntent sender = PendingIntent.getBroadcast(context, alarmId, myIntent, 0);
AlarmManager alarm = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
alarm.set(AlarmManagerType, cycTime, sender);
}
/** * 取消闹钟 * * @param context * @param action */
public static void cancelAlarmTimer(Context context, String action) {
Intent myIntent = new Intent();
myIntent.setAction(action);
PendingIntent sender = PendingIntent.getBroadcast(context, 0, myIntent,0);
AlarmManager alarm = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
alarm.cancel(sender);
}
}复制代码