AlarmManager使用

AlarmManager是Android中常用的一种系统级别的提示服务,在特定的时刻为我们广播一个指定的Intent,为你的应用设定一个在未来某个时间唤醒的功能。

当闹钟响起,实际上是系统发出了为这个闹钟注册的广播,会自动开启目标应用。

注册的闹钟在设备睡眠的时候仍然会保留,可以选择性地设置是否唤醒设备,但是当设备关机和重启后,闹钟将会被清除。

对于常规的短时间计时操作(ticks, timeouts, etc),使用Handler处理更加方便和有效率。


alarm的receiver的onReceive()方法被执行的时候,AlarmManager持有一个CPU唤醒锁,这样就保证了设备在处理完广播之前不会sleep。一旦onReceive()方法返回,AlarmManager就会释放这个锁,表明一些情况下可能onReceive()方法一执行完设备就会sleep。如果你的alarmreceiver中调用了Context.startService(),那么很可能service还没起来设备就sleep了。为了阻止这种情况,你的BroadcastReceiver和Service需要实现不同的唤醒锁机制,来确保设备持续运行到service可用为止。


如果onReceive()方法里确实需要异步操作的话,可以用goAsync方法,然后在新开一个线程去执行。

如果onReceive()有过于耗时的操作,建议使用PendingIntent.getService()启动Service方式来完成。

BroadcastReceiver onReceive()使用goAsync 执行异步操作例子:

@Overridepublic void onReceive(final Context context, final Intent intent) {final PendingResult result = goAsync();final PowerManager.WakeLock wl = AlarmAlertWakeLock.createPartialWakeLock(context);try {wl.acquire();} catch (Exception e) {e.printStackTrace();}AsyncHandler.post(new Runnable() {@Overridepublic void run() {LogUtils.writeAlarmLog("AlarmStateManager received intent " + intent);try {handleIntent(context, intent);result.finish();wl.release();} catch (Exception e) {e.printStackTrace();}}});}

Alarm接口

1.cancel(PendingIntent operation)

 Remove any alarms with a matching Intent.

2.changeAlarmType(String pkgName, boolean wakeup)

3.getNextAlarmClock()

Gets information about the next alarm clock currently scheduled.

4.set(int type, long triggerAtMillis, PendingIntent operation)

Schedule an alarm.

5.setAlarmClock(AlarmManager.AlarmClockInfo info, PendingIntent operation)

Schedule an alarm that represents an alarm clock.

6.setExact(int type, long triggerAtMillis, PendingIntent operation)

Schedule an alarm to be delivered precisely at the stated time.

7.setInexactRepeating(int type, long triggerAtMillis, long intervalMillis, PendingIntent operation)

Schedule a repeating alarm that has inexact trigger time requirements; for example, an alarm that repeats every hour, but not necessarily at the top of every hour.

8.setRepeating(int type, long triggerAtMillis, long intervalMillis, PendingIntent operation)

Schedule a repeating alarm.

9.setTime(long millis)

Set the system wall clock time.

10.setTimeZone(String timeZone)

Set the system default time zone.

11.setWindow(int type, long windowStartMillis, long windowLengthMillis, PendingIntent operation)

Schedule an alarm to be delivered within a given window of time.


常用接口

1.public void set/setExact(int type, long triggerAtMillis, PendingIntent operation)

 该方法用于设置一次性闹钟。

第一个参数int type 指定定时服务的类型,该参数接受如下值:

ELAPSED_REALTIME: 在指定的延时过后,发送广播,但不唤醒设备(闹钟在睡眠状态下不可用)。如果在系统休眠时闹钟触发,它将不会被传递,直到下一次设备唤醒。

ELAPSED_REALTIME_WAKEUP: 在指定的延时过后,发送广播,并唤醒设备(即使关机也会执行operation所对应的组件)。延时是要把系统启动的时间SystemClock.elapsedRealtime()算进去的,具体用法看代码。

RTC: 指定当系统调用System.currentTimeMillis()方法返回的值与triggerAtTime相等时启动operation所对应的设备(在指定的时刻,发送广播,但不唤醒设备)。如果在系统休眠时闹钟触发,它将不会被传递,直到下一次设备唤醒(闹钟在睡眠状态下不可用)。

RTC_WAKEUP: 指定当系统调用System.currentTimeMillis()方法返回的值与triggerAtTime相等时启动operation所对应的设备(在指定的时刻,发送广播,并唤醒设备)。即使系统关机也会执行 operation所对应的组件。

第二个参数triggerAtMillis表示触发闹钟的时间。

第三个参数PendingIntent pi表示闹钟响应动作:

PendingIntent pi:是闹钟的执行动作,比如发送一个广播、给出提示等等。PendingIntent是Intent的封装类。需要注意的是:

启动服务:如果是通过启动服务来实现闹钟提示的话,PendingIntent对象的获取就应该采用Pending.getService(Context c,int i,Intent intent,int j)方法;

启动广播:如果是通过广播来实现闹钟提示的话,PendingIntent对象的获取就应该采用PendingIntent.getBroadcast(Context c,inti,Intent intent,int j)方法;

启动activity:如果是采用Activity的方式来实现闹钟提示的话,PendingIntent对象的获取就应该采用PendingIntent.getActivity(Context c,inti,Intent intent,int j)方法。

如果这三种方法错用了的话,虽然不会报错,但是看不到闹钟提示效果。

注意:AndroidL开始,设置的alarm的触发时间必须大于当前时间 5秒

AlarmManagerService中是通过PendingItent来标示一个Alarm的

2.public void cancel (PendingIntent operation)

移除intent相匹配的alarm(只要action,data, type,package,component,categories相等,就会被取消)。另外,应用被forcestoppackage杀掉之后,所有的alarm都会被移除。

3.public void setRepeating(int type,long startTime,long intervalTime,PendingIntent pi)

设置一个周期性执行的定时服务。第一个参数表示闹钟类型,第二个参数表示闹钟首次执行时间,第三个参数表示闹钟两次执行的间隔时间,第三个参数表示闹钟响应动作。

注意:该方法提供了设置周期闹钟的入口,闹钟执行时间严格按照startTime来处理,使用该方法需要的资源更多,不建议使用。

4.public void setInexactRepeating(int type,long startTime,long intervalTime,PendingIntent pi)

该方法也用于设置重复闹钟,与第二个方法相似,不过其两个闹钟执行的间隔时间不是固定的而已。它相对而言更省电(power-efficient)一些,因为系统可能会将几个差不多的闹钟合并为一个来执行,减少设备的唤醒次数。第三个参数intervalTime为闹钟间隔,内置的几个变量如下:

INTERVAL_DAY: 设置闹钟,间隔一天

INTERVAL_HALF_DAY: 设置闹钟,间隔半天

INTERVAL_FIFTEEN_MINUTES:设置闹钟,间隔15分钟

INTERVAL_HALF_HOUR: 设置闹钟,间隔半个小时

INTERVAL_HOUR: 设置闹钟,间隔一个小时

AndroidL开始repeat的周期必须大于60秒


设置一个闹钟和取消一个闹钟的例子:

private static class AlarmManagerStateChangeScheduler implements StateChangeScheduler {@Overridepublic void scheduleInstanceStateChange(Context context, Calendar time,AlarmInstance instance, int newState) {long timeInMillis = time.getTimeInMillis();Intent stateChangeIntent = createStateChangeIntent(context, ALARM_MANAGER_TAG, instance,newState, VALUE_CHANGE_STATE_BY_SCHEDULE);// Treat alarm state change as high priority, use foreground broadcastsstateChangeIntent.addFlags(Intent.FLAG_RECEIVER_FOREGROUND);PendingIntent pendingIntent = PendingIntent.getService(context, instance.hashCode(),stateChangeIntent, PendingIntent.FLAG_UPDATE_CURRENT);//设置一个闹钟AlarmManager am = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);if (CommonUtils.isMOrLater()) {am.setExactAndAllowWhileIdle(ReflectUtils.getRTC_BOOT_WAKEUP()/*AlarmManager.RTC_WAKEUP*/, timeInMillis, pendingIntent);LogUtils.writeAlarmLog("Scheduling state change success by setExactAndAllowWhileIdle");} else if (CommonUtils.isKitKatOrLater()) {am.setExact(ReflectUtils.getRTC_BOOT_WAKEUP()/*AlarmManager.RTC_WAKEUP*/, timeInMillis, pendingIntent);LogUtils.writeAlarmLog("Scheduling state change success by setExact");} else {am.set(AlarmManager.RTC_WAKEUP, timeInMillis, pendingIntent);LogUtils.writeAlarmLog("Scheduling state change success not KitKatOrLater");}}@Overridepublic void cancelScheduledInstanceStateChange(Context context, AlarmInstance instance) {LogUtils.writeAlarmLog("cancelScheduledInstance instance id = " + instance.mId);// Create a PendingIntent that will match any one set for this instancePendingIntent pendingIntent = PendingIntent.getService(context, instance.hashCode(),createStateChangeIntent(context, ALARM_MANAGER_TAG, instance, null, null),PendingIntent.FLAG_NO_CREATE);//取消一个闹钟if (pendingIntent != null) {AlarmManager am = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);am.cancel(pendingIntent);pendingIntent.cancel();ReflectUtils.cancelOnePoweroffAlarm(am, pendingIntent);}}}


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/430307.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

controller调用controller的方法_SpringCloud(5):Feign整合Ribbon和Hystrix来进行远程调用与服务熔断...

Feign前面我们虽然使用了Ribbon和Hystrix进行了负载均衡和服务熔断,但是我们的消费方代码中由于要调用提供方的服务,比如:String user this.restTemplate.getForObject("http://service-provider/user/" id, String.class);使用了…

w3svc无法启动

w3svc无法启动 运行命令regedit,打开注册表编辑器,进入:HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\HTTP]: a) Change Start from 4 ( disabled) to 3 ( automatically)b) Delete "NoRun" key if this key exists 重启…

Retrofit的简单使用

Retrofit是REST安卓客户端请求库。使用Retrofit可以进行GET&#xff0c;POST&#xff0c;PUT&#xff0c;DELETE等请求方式。官方文档&#xff1a;http://square.github.io/retrofit/ 1.在AndroidManifest.xml中请求了网络权限&#xff1a; <uses-permission android:name&…

vba excel 退出编辑状态_VBA小常识(15)—Application对象

VBA小常识&#xff08;15&#xff09;—Application对象学习自杨洋老师《全民一起学VBA》Application是位于最顶端的对象&#xff0c;代表excel程序本身&#xff0c;所有对象都属于它。1. application.cells当前处于激活状态的工作表&#xff0c;即“活动工作表”的单元格。此时…

手机--装饰者模式

/* * 装饰者* 装饰者与被装饰对象有相同的父类* 装饰者内包含被装饰者对象的引用* * 组合替代继承来扩展对象行为&#xff0c;运行时根据需求动态扩展* 扩展开放&#xff0c;修改关闭。* */public class Phone { }public class IPhone : Phone{public IPhone() { Console.Write…

一个网站 Code/Android/

http://www.java2s.com/Code/Android/

c++ stack 遍历_划重点啦!带你解读图的两种遍历方式

01知识框架02图的遍历1深度优先遍历基本思想&#xff1a;首先访问图中起始顶点v&#xff0c;然后由v出发&#xff0c;访问与v邻接且未被访问的顶点再访问与v相邻且未被访问的顶点 w1...重复上述过程。当不能再继续向下访问时&#xff0c;依次退回到最近被的问的顶点&#xff0c…

for循环延时_单片机的独立按键学习,实现60秒循环数字的启动暂停与清零

想要学习单片机必须要准备单片机开发板一块&#xff0c;和单片机C语言版本的书籍&#xff0c;我会把每天录制视频的源代码到头条&#xff0c;并且录制相同文章名字视频到西瓜视频&#xff0c;如果你再学习独立按键这一块有不明白的地方可以点开我的头像&#xff0c;进入我的主页…

本来中午打算应付下随便吃点,可是连盐都没有放的辣椒炒蛋实在是令人不快...

可是现在由于这个问题已经产生&#xff0c;我就将计就计&#xff0c;把他的空调电池给弄了&#xff0c;额后下午起来已经是7点半&#xff0c;就出去买了一袋4元的盐&#xff0c;本来还是打算会醒来早点&#xff0c;可是自己到了夜间才睡醒&#xff0c;还是将就再吃个没有放辣椒…

调用startactivityforresult后,onactivityresult立刻返回RESULT_CANCELED

调用startActivityForResult后&#xff0c;onActivityResult立刻响应&#xff0c;原因如下&#xff1a;1. 去掉跳转到的页面的android:launchMode"singleTask"属性&#xff08;SINGLE_TASK标识以及SINGLE_INSTANCE两个标识必须在r.result0的条件中&#xff0c;即这两…

mysql创建全外连接的视图_关系型数据库 MySQL 表索引和视图详解

原创&#xff1a; JiekeXu JiekeXu之路 一、索引数据库索引通俗的讲就是和书本的目录一样&#xff0c;主要就是为了提高查询数据的效率。由于数据存储在数据库表中&#xff0c;所以索引是创建在数据库表对象上&#xff0c;由表中的一个字段或多个字段生成的键组成&#xff0c;这…

FatMouse's Speed hdu 1160(动态规划,最长上升子序列+记录路径)

http://acm.hdu.edu.cn/showproblem.php?pid1160 题意&#xff1a;现给出老鼠的体重与速度&#xff0c;要求你找出符合要求的最长子序列。 要求是 W[m[1]] < W[m[2]] < ... < W[m[n]]&#xff08;体重&#xff09; && S[m[1]] > S[m[2]] > ... > S[…

设计一个名为complex的类来表示复数_complex类java解决

作者&#xff1a;wzu_cza123出自&#xff1a;CSDN原文&#xff1a;blog.csdn.net/wzu_cza123/article/details/108857953?utm_mediumdistribute.pc_category.none-task-blog-hot-5.nonecase&depth_1-utm_sourcedistribute.pc_category.none-task-blog-hot-5.nonecase&…

java将经纬度转换为平面坐标

&#xff08;米勒投影&#xff09;算法public static double[] MillierConvertion(double lat, double lon) { double L 6381372 * Math.PI * 2;//地球周长 double WL;// 平面展开后&#xff0c;x轴等于周长 double HL/2;// y轴约等于周长一半 double mill2.3;// 米勒投影…

小程序onload_微信小程序开发入门之共享账本(十四)

微信小程序开发入门之共享账本(十四)(备注&#xff1a;微信小程序的wxml文件相当于HTML文件&#xff0c;wxss文件相当于CSS文件&#xff0c;js文件就是JavaScript文件&#xff0c;数据库为NoSQL数据库&#xff0c;数据库脚本语言也同NoSQL&#xff0c;因为是运行在微信内&#…

POJ 3342 树形DP+Hash

这是很久很久以前做的一道题&#xff0c;可惜当时WA了一页以后放弃了。 今天我又重新捡了起来。&#xff08;哈哈1A了&#xff09; 题意&#xff1a; 没有上司的舞会判重 思路&#xff1a; hash一下树形DP 题目中给的人名hash到数字&#xff0c;再进行运算。 树形DP f[x…

Ubuntu 安装 Vue.js

1 配置Node (1) 从https://nodejs.org/en/download/ 下载node.js&#xff0c;例如 node-v6.11.1-linux-x64.tar.xz (2) 解压到一个地方&#xff0c;然后配置全局变量&#xff0c;在/etc/profile 添加下面配置信息&#xff1a; export NODE_HOME/home/junliang/Soft/node-v6.11.…

MyISAM表加字段的特殊方法

最近一个统计系统的大表需要加字段&#xff0c;表的引擎是myisam&#xff0c;表大小在3亿&#xff0c;物理文件在106G。想想都蛋疼。那么这种情况下怎么把字段撸上去呢? 1. 首先想到了《高性能MySQL》提到的直接更改表结构文件&#xff08;frm&#xff09;&#xff0c;但是在经…

python正则表达式面试题_吐血总结!50道Python面试题集锦(4)

Q31、如何在python中注释多行&#xff1f; 注释多行代码时。所有要注释的行都要在开头前加#。还可以使用快捷方式来注释多行&#xff0c;就是按住Ctrl键并在每个想要包含&#xff03;字符的地方左键单击并键入一次&#xff03;。 Q32、什么是Python中的文档Docstrings&#xff…

iOS本地推送UILocalNotification

本地通知主要是基于app本身定时器的行为。即使app在后台&#xff0c;也会发送本地通知。一个app只能有有限数量的预定通知&#xff0c;最多允许最近的64条通知&#xff0c;其余通知将会被系统忽略。 推送通知的呈现效果: 在屏幕顶部显示的一条横幅在屏幕中间弹出一个UIAlertVie…