当前位置: 首页 > news >正文

关于Android中设置闹钟的相对比较完善的解决方案

前些时候,有人在我「非著名程序员」公众号的后台问我有没有设置闹钟的demo,我当时说承诺为大家写一个,一直没空,直到最近又有人跟我要,我决定抽时间写一个吧。确实设置闹钟是一个比较麻烦的东西。我在这里写的这个demo抽出来了封装了一个类库,大家直接调用其中的设置闹钟和取消闹钟的方法即可。可以设置每天的闹钟,周一到星期天之间多选的闹钟,也可以设置选择一次性闹钟,跟系统设置闹钟的方式基本差不多吧。

效果图

来看分析和讲解之前,先看看效果吧,效果图如下:

设置界面

闹钟提醒界面

AlarmManager

对于AlarmManager里的方法我就不逐一介绍了,如果都介绍讲完估计就天黑了。AlarmManager这个类提供对系统闹钟服务的访问接口。
在API 19 以前,AlarmManager的常用方法有三个:

  • set(int type,long startTime,PendingIntent pi);
    该方法用于设置一次性闹钟,第一个参数表示闹钟类型,第二个参数表示闹钟执行时间,第三个参数表示闹钟响应动作。

  • setRepeating(int type,long startTime,long intervalTime,PendingIntent pi);
    该方法用于设置重复闹钟,第一个参数表示闹钟类型,第二个参数表示闹钟首次执行时间,第三个参数表示闹钟两次执行的间隔时间,第三个参数表示闹钟响应动作。

  • setInexactRepeating(int type,long startTime,long intervalTime,PendingIntent pi);
    该方法也用于设置重复闹钟,与第二个方法相似,不过其两个闹钟执行的间隔时间不是固定的而已。

从API 19开始,AlarmManager的机制都是非准确传递,操作系统将会转换闹钟,来最小化唤醒和电池使用。targetSdkVersion在API 19之前应用仍将继续使用以前的行为,所有的闹钟在要求准确传递的情况下都会准确传递。

从API 19以后,则采用了如下方法:

  • setWindow(int, long, long, PendingIntent)

  • setExact(int, long, PendingIntent)
    从上面的两个方法我们可以看出,没有了repeat,就是设置了闹钟只能响一次了,而且这两种方法都可以设置精确的,第一个相对于第二种方法来说,应该是比较省电的。因为setWindow这个方法允许应用程序利用电池优化来自交货批处理即使它适度的及时性要求警报。

主要问题

  1. API 19以后没有了重复设置,那如果设置一个闹钟每天都准确提醒呢?

  2. 手机重启之后,设置的闹钟是否还有效?

  3. 应用程序被杀死之后,闹钟是否还有效?

说实话,这些问题我相信大家肯定都遇到过,而且解决起来相当费劲,确实是。来看我们如何一一解决吧。

解决遇到的坑

API 19以后如何设置重复闹钟

我们知道,我们在使用AlarmManager设置了提醒之后,是通过广播接收的,设置的提醒时间一到,系统发送我们自定义的广播,我们接收到,应用程序提醒。那提醒的时候,我们可以再重新设置一次嘛,这就解决了API 19设置重复闹钟的问题。

PendingIntent sender = PendingIntent.getBroadcast(context, id, intent, PendingIntent
                .FLAG_CANCEL_CURRENT);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            am.setWindow(AlarmManager.RTC_WAKEUP, calMethod(week, calendar.getTimeInMillis()),
                    intervalMillis, sender);
        } else {
            if (flag == 0) {
                am.set(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), sender);
            } else {
                am.setRepeating(AlarmManager.RTC_WAKEUP, calMethod(week, calendar.getTimeInMillis
                        ()), intervalMillis, sender);
            }
        }

根据判断系统版本,使用不同的设置闹钟的方法,进行设置。接下来我们通过广播接收系统发来的通知,进行闹钟提醒。

public class LoongggAlarmReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        // TODO Auto-generated method stub
        String msg = intent.getStringExtra("msg");
        long intervalMillis = intent.getLongExtra("intervalMillis", 0);
        if (intervalMillis != 0) {
            AlarmManagerUtil.setAlarmTime(context, System.currentTimeMillis() + intervalMillis,
                    intent);
        }
        int flag = intent.getIntExtra("soundOrVibrator", 0);
        Intent clockIntent = new Intent(context, ClockAlarmActivity.class);
        clockIntent.putExtra("msg", msg);
        clockIntent.putExtra("flag", flag);
        clockIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        context.startActivity(clockIntent);
    }
}

通过上面的广播,我们可以看到,我是通过那个时间间隔是否为零来判断API 19之后是否是重复闹钟,不为0,就再重新设置一遍。我们来一起看看setAlarmTime()这个方法。如下:

 public static final String ALARM_ACTION = "com.loonggg.alarm.clock";

    public static void setAlarmTime(Context context, long timeInMillis, Intent intent) {
        AlarmManager am = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
        PendingIntent sender = PendingIntent.getBroadcast(context, intent.getIntExtra("id", 0),
                intent, PendingIntent.FLAG_CANCEL_CURRENT);
        int interval = (int) intent.getLongExtra("intervalMillis", 0);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            am.setWindow(AlarmManager.RTC_WAKEUP, timeInMillis, interval, sender);
        }
    }

就这样,重复的问题就解决了。

手机重启之后,闹钟失效怎么解决

对,手机重启之后,闹钟确实是失效了,要想解决这个问题,那就再设置一个监听手机重启的广播,等手机重启的时候,再重新设置一遍,即可解决上面的问题。

看看我在手机重启广播里调用了我封装的重新设置闹钟的方法。这样就解决了手机重启之后,闹钟失效的问题。
注意:广播需要在清单文件注册,小伙伴们别忘记了哈。我在这里就不贴出代码来了。

应用程序被杀死,闹钟失效

可以告诉你们的是,这个还真的没有什么好的解决方案,如果你们程序里写了服务,可以在服务重启的时候,判断重新再把闹钟注册一次,或者在打开应用的时候重新注册一次,反正就是能有利于闹钟注册的地方,进行重新设置。如果闹钟设置的id是一样的,后边设置的会自动覆盖先前设置的闹钟。如果谁有比较好的解决应用程序被杀死后,闹钟失效的问题,欢迎大家提供出来分享。

这个封装的类库的好处

好处就是我把方法都给你们封装好了,直接就可以调用。

  • 直接传入时分的值就可以了。比如:直接传入某个时间点:12:30,然后传入是否是每天提醒,还是周几提醒等

  • 闹钟提醒的界面我也已经封装到里面了,还算好看,懒的同学不需要再写了,不满意的同学可以直接下载类库修改。

  • 取消闹钟的方法,我也已经进行了封装。

总之,就是非常方便,到底有多方便大家直接看demo就知道了,不满意的同学可以直接下载类库进行修改。

demo和类库地址:https://github.com/loonggg/Android-AlarmManagerClock

移动开发者的聚集地,公众号“非著名程序员”,每天一篇原创技术分享和移动互联网知识分享,微信公众号:smart_android。

相关文章:

  • Swift 中的尾递归和蹦床
  • 关于学习了《构建之法》的若干存在疑惑的问题
  • spring 整合 struts
  • open-falcon+grafana出图
  • C语言应用-飞机游戏
  • 第二阶段冲刺第八天
  • Servlet与JSP的区别
  • swift学习笔记-UI篇之UIImageView
  • 我有几个粽子,和一个故事
  • openstack dnsmasq彭祖
  • WSS与Project Server集成 - Chatterley - 博客园
  • mysql 5.6 原生Online DDL解析
  • 第二阶段个人总结11
  • HDU 2073 无限的路
  • kettle 数据迁移 (转)
  • 【vuex入门系列02】mutation接收单个参数和多个参数
  • 【跃迁之路】【444天】程序员高效学习方法论探索系列(实验阶段201-2018.04.25)...
  • Centos6.8 使用rpm安装mysql5.7
  • css系列之关于字体的事
  • CSS中外联样式表代表的含义
  • Cumulo 的 ClojureScript 模块已经成型
  • java架构面试锦集:开源框架+并发+数据结构+大企必备面试题
  • JS+CSS实现数字滚动
  • mysql 数据库四种事务隔离级别
  • 第三十一到第三十三天:我是精明的小卖家(一)
  • 驱动程序原理
  • 如何使用Mybatis第三方插件--PageHelper实现分页操作
  • 入手阿里云新服务器的部署NODE
  • NLPIR智能语义技术让大数据挖掘更简单
  • Redis4.x新特性 -- 萌萌的MEMORY DOCTOR
  • 如何用纯 CSS 创作一个菱形 loader 动画
  • $.extend({},旧的,新的);合并对象,后面的覆盖前面的
  • (2009.11版)《网络管理员考试 考前冲刺预测卷及考点解析》复习重点
  • (23)Linux的软硬连接
  • (solr系列:一)使用tomcat部署solr服务
  • (zhuan) 一些RL的文献(及笔记)
  • (附源码)springboot太原学院贫困生申请管理系统 毕业设计 101517
  • (附源码)计算机毕业设计SSM疫情下的学生出入管理系统
  • (三) prometheus + grafana + alertmanager 配置Redis监控
  • (三)elasticsearch 源码之启动流程分析
  • (万字长文)Spring的核心知识尽揽其中
  • (原創) 如何將struct塞進vector? (C/C++) (STL)
  • (转)Android学习笔记 --- android任务栈和启动模式
  • (转)大型网站的系统架构
  • (转)我也是一只IT小小鸟
  • (转载)Linux网络编程入门
  • ..thread“main“ com.fasterxml.jackson.databind.JsonMappingException: Jackson version is too old 2.3.1
  • ./和../以及/和~之间的区别
  • .NET/C# 使用反射注册事件
  • [100天算法】-每个元音包含偶数次的最长子字符串(day 53)
  • [BROADCASTING]tensor的扩散机制
  • [C/C++]数据结构 深入挖掘环形链表问题
  • [LLM][FT]大模型Fine-Tuning相关技术0
  • [LOJ161] 仙人掌计数
  • [Oh My C++ Diary]用cout输出流保留输出小数位数的方法