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

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

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

(0)

相关推荐

  • Android开发之使用通知栏显示提醒信息的方法

    本文实例讲述了Android开发之使用通知栏显示提醒信息的方法.分享给大家供大家参考,具体如下: 用通知栏来提醒 public void notifyKJ() { //获得通知管理器,通知是一项系统服务 NotificationManager manager = (NotificationManager) context.getSystemService(context.NOTIFICATION_SERVICE); //初始化通知对象 p1:通知的图标 p2:通知的状态栏显示的提示 p3:通知显

  • 详解Android中Notification通知提醒

    在消息通知时,我们经常用到两个组件Toast和Notification.特别是重要的和需要长时间显示的信息,用Notification就最 合适不过了.当有消息通知时,状态栏会显示通知的图标和文字,通过下拉状态栏,就可以看到通知信息了,Android这一创新性的UI组件赢得了用户的一 致好评,就连苹果也开始模仿了.今天我们就结合实例,探讨一下Notification具体的使用方法.  首先说明一下我们需要实现的功能是:在程序启动时,发出一个通知,这个通知在软件运行过程中一直存在,相当于qq的托盘

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

    AlarmManager简介 AlarmManager实质是一个全局的定时器,是Android中常用的一种系统级别的提示服务,在指定时间或周期性启动其它组件(包括Activity,Service,BroadcastReceiver).本文将讲解一下如何使用AlarmManager实现定时提醒功能. 闹钟配置 周期闹钟 Intent intent = new Intent(); intent.setAction(GlobalValues.TIMER_ACTION_REPEATING); Pendi

  • Android开发之Notification手机状态栏通知用法实例分析

    本文实例讲述了Android开发之Notification手机状态栏通知用法.分享给大家供大家参考,具体如下: 简介: 通知是显示在手机状态栏的通知(PS:就是手机上方,显示时间啥的那一栏) 用法: Notification添加了Builder()类,其包含如下方法: 1. setDefaults()         通知led灯.音乐.震动等 2. setAutoChange()  设置点击通知后,通知自动从状态栏删除 3. setContentTitle()   通知标题 4. setCon

  • Android中AlarmManager基本用法分析

    本文实例讲述了Android中AlarmManager基本用法.分享给大家供大家参考,具体如下: AlarmManager的作用文档中的解释是:在特定的时刻为我们广播一个指定的Intent.简单的说就是我们设定一个时间,然后在该时间到来时,AlarmManager为我们广播一个我们设定的Intent. 对应AlarmManager更深层的了解可以参考: http://www.jb51.net/article/90491.htm android提供了四种类型的闹钟: ① ELAPSED_REALT

  • Android中new Notification创建实例的最佳方法

    目前 Android 已经不推荐使用下列方式创建 Notification实例: Notification notification = new Notification(R.drawable.ic_launcher,"This is ticker text",System.currentTimeMillis()); 最好采用下列方式: Notification notification = new Notification.Builder(this) .setContentTitle

  • Android中RecyclerView拖拽、侧删功能的实现代码

    废话不多说,下面展示一下效果. 这是GridView主文件实现. public class GridViewActivity extends AppCompatActivity { RecyclerView mRecyclerView; List<String> mStringList; RecyclerAdapter mRecyAdapter; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { s

  • Android中imageView图片放大缩小及旋转功能示例代码

    一.简介 二.方法 1)设置图片放大缩小效果 第一步:将<ImageView>标签中的android:scaleType设置为"fitCenter" android:scaleType="fitCenter" 第二步:获取屏幕的宽度 DisplayMetrics dm=new DisplayMetrics(); getWindowManager().getDefaultDisplay().getMetrics(dm); dm.widthPixels 第三

  • Android中的Bmob移动后端云服务器功能

    源码下载:http://xiazai.jb51.net/201801/yuanma/BmobTest1.rar PS:一般情况下,我们在写android程序的时候,想要实现登录注册功能,可以选择自己用servlet作为服务端来实现过滤没有注册过的用户,但是太麻烦,而且不是随时都可以用的.这里介绍一个移动后端云服务器平台bmob,这不仅可以实现云数据库储存,还可以获取手机验证等,随时随地都很轻松,下面写一个小demo,实现一个登陆注册功能,认识增删查改.下面我稍微写一个例子,简单实现注册登录功能.

  • Android中BroadcastReceiver实现短信关键字自动回复功能

    前言:因公司业务需要,需要一台手机专门回复客户订购的套餐的短信,之前一直是人工手动回复,但比较麻烦且回复可能不及时,于是项目经理就让实现根据短信的关键字自动回复功能. 实现手机短信监听的方式有两种:一是通过ContentObserver观察者实现监听,另一种就是通过广播即BroadcastReceiver实现短信监听,文章中通过使用BroadcastReceiver实现有新短信的及时监听及包含设定的关键字时自动回复. 效果图: 清单文件添加权限: <uses-permission android

  • Android中关于Notification及NotificationManger的详解

    Android状态栏提醒 在Android中提醒功能也可以用AlertDialog,但是我们要慎重的使用,因为当使用AlertDialog的时候,用户正在进行的操作将会被打断,因为当前焦点被AlertDialog得到.我们可以想像一下,当用户打游戏正爽的时候,这时候来了一条短信.如果这时候短信用AlertDialog提醒,用户必须先去处理这条提醒,从而才能继续游戏.用户可能会活活被气死.而使用Notification就不会带来这些麻烦事,用户完全可以打完游戏再去看这条短信.所以在开发中应根据实际

  • Android中FTP上传、下载的功能实现(含进度)

    Android中使用的FTP上传.下载,含有进度. 代码部分主要分为三个文件:MainActivity,FTP,ProgressInputStream 1. MainActivity package com.ftp; import java.io.File; import java.io.IOException; import java.util.LinkedList; import com.ftp.FTP.DeleteFileProgressListener; import com.ftp.F

随机推荐