欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android中AlarmManager+Notification实现定时通知提醒功能

程序员文章站 2023-12-13 12:46:10
alarmmanager简介 alarmmanager实质是一个全局的定时器,是android中常用的一种系统级别的提示服务,在指定时间或周期性启动其它组件(包括acti...

alarmmanager简介

alarmmanager实质是一个全局的定时器,是android中常用的一种系统级别的提示服务,在指定时间或周期性启动其它组件(包括activity,service,broadcastreceiver)。本文将讲解一下如何使用alarmmanager实现定时提醒功能。

闹钟配置

周期闹钟

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:闹钟响应动作。

setinexactrepeating(int type,long starttime,long intervaltime,pendingintent pi)

该方法也用于设置周期定式服务,与上一种类似。不过其两个闹钟执行的间隔时间不是固定的。它相对而言更省电一些,因为系统可能会将几个差不多的闹钟合并为一个来执行,减少设备的唤醒次数。

intervaltime内置变量

间隔一天:   interval_day
间隔半天:   interval_half_day
间隔15分钟:  interval_fifteen_minutes
间隔半个小时: interval_half_hour
间隔一个小时: interval_hour

定时闹钟

//获得系统提供的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唯一性。

 //给每个闹钟设置不同id防止覆盖
int alarmid = sharedpreutils.getinteger(context, "alarm_id", 0);
sharedpreutils.setinteger(context, "alarm_id", ++alarmid);
pendingintent sender = pendingintent.getbroadcast(context, alarmid, myintent, 0);

在取消闹钟时我们也可以根据这个id关闭不同的闹钟。

参数详解

type:闹钟类型

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版本影响,某些版本并不支持。

long intervaltime:执行时间

闹钟的第一次执行时间,以毫秒为单位,可以自定义时间,不过一般使用当前时间。需要注意的是,本属性与第一个属性(type)密切相关,如果第一个参数对应的闹钟使用的是相对时间(elapsed_realtime和elapsed_realtime_wakeup),那么本属性就得使用相对时间(相对于系统启动时间来说),比如当前时间就表示为:systemclock.elapsedrealtime();如果第一个参数对应的闹钟使用的是绝对时间(rtc、rtc_wakeup、power_off_wakeup),那么本属性就得使用绝对时间,比如当前时间就表示为:system.currenttimemillis()

long starttime:间隔时间

对于周期定时方式来说,存在本属性,表示两次闹钟执行的间隔时间,也是以毫秒为单位。

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)方法。如果这三种方法错用了的话,虽然不会报错,但是看不到闹钟提示效果。。

广播配置

新建闹钟broadcastreceiver:

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
  }
 }
}

注册broadcastreceiver:

最后别忘了在清单里注册广播。

<!--闹钟接收广播-->
<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);
 }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

上一篇:

下一篇: