美文网首页Android非著名程序员UI & Material Design
关于Android中设置闹钟的相对完善的解决方案

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

作者: 非著名程序员 | 来源:发表于2016-05-25 17:23 被阅读7591次

前些时候,有人在我「非著名程序员」微信公众号的后台问我有没有设置闹钟的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

相关文章

网友评论

  • tmyzh:请问下博主,就是程序杀死之后闹钟的功能,有做什么保活的措施吗
  • be757fadea9d:请问,设定每天重复的闹钟之后,手动调节日期到明天定时之前,闹钟会提示 有什么解决思路吗?
  • e416c0064146:请问楼主:我APP中用到闹钟,直接把您示例文件中的AlarmManagerUtil.java拖到我项目中,setAlarm方法设置闹钟为什么无效呢?还需要别的文件吗?
    e416c0064146:问题解决,Android studio中,广播注册设置错误,应该是
    <receiver android:name=".LoongggAlarmReceiver"
    android:enabled="true"
    android:exported="true">
    <intent-filter>
    <action android:name="com.loonggg.alarm.clock" />
    </intent-filter>
    </receiver>
    action android:name应该是intent 的名字。
    e416c0064146:AlarmManagerUtil.java, LoongggAlarmReceiver.java, ClockAlarmActivity.java, SimpleDialog.java都添加到项目里了,闹钟设置代码:
    AlarmManagerUtil.setAlarm(MineActivity.this, 0, 15, 20, 1, 0, "testest", 2);
    SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    Date date = null;
    try {
    date = formatter.parse("2017-09-18 11:21:00");
    } catch (Exception e) {

    }
    Intent intent = new Intent(MineActivity.ALARM_SERVICE);
    AlarmManagerUtil.setAlarmTime(MineActivity.this, date.getTime(), intent);

    还是无效。
    e416c0064146:另外三个文件也添加到项目里面了,还是没有闹钟,<receiver android:name=".LoongggAlarmReceiver" >
    <intent-filter android:priority="1000" >
    <action android:name="com.test.AlarmManagerUtil" />
    </intent-filter>
    </receiver>
    这个是广播注册的文件,我用的是华为的手机测试的。
  • 彩色浪花之子:我想说关于重启手机失效的问题,你这并没有监听手机重启的事件啊,
    e416c0064146:我也是没有解决,按照网上说的设置了,也没有用。
    <receiver android:name=".LoongggAlarmReceiver"
    android:enabled="true"
    android:exported="true">
    <intent-filter>
    <action android:name="android.intent.action.BOOT_COMPLETED" />
    </intent-filter>
    </receiver>
    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"></uses-permission>
    都设置了,也不行,如果你解决了,可否发上来一下?
  • Newamber:楼主,我想问问,你那个AlarmManagerUtil里面有这么一行代码
    calendar.set(calendar.get(Calendar.YEAR), calendar.get(Calendar.MONTH), calendar.get
    (Calendar.DAY_OF_MONTH), hour, minute, 10);

    我知道这是设置用户输入的真实时间,并在getTimeInMillis()中以毫秒的格式返回,但是 calendar.get(Calendar.MONTH),不是应该加一吗,因为比如现在是三月,调用它会返回2。。。不是很懂,然而demo是会按时响起的,照理说,应该是设置的上一个月的同时刻闹钟啊
  • 3a6723356cae:请问楼主。闹钟延时怎么解决?基本都在两分钟左右
    超神的菠萝:你好,2分钟延时那个问题解决了吗~~
    超神的菠萝:我遇到2分钟延迟,还有不响的问题啊。。。不需要放在servic里面吗?切到后台就不响了。。
    非著名程序员:@zTtao 延时没有什么好的解决方案。
  • 小邪正吕:请问一下楼主, 关于设置闹钟 偶发性失效有解决方法吗
  • 我好芒:请问一下楼主,为啥调用你那个工具类的cancelAlarm()方法怎么取消不了闹钟?
    Wang_Guan:@wyu_xks 非常感谢!:smile:
    我好芒:@不调皮的粉_Orz 解决了,把 am.setWindow(AlarmManager.RTC_WAKEUP, timeInMillis, interval, sender)换成这个就可以了am.setExact(AlarmManager.RTC_WAKEUP, calMethod(week, calendar.getTimeInMillis()), sender),然后取消的时候根据闹钟id取消对应的闹钟。
    Wang_Guan:请问解决了吗?我也遇到取消不了:joy:
  • ttdevs:标题不错
  • 一个盖世英雄:为什么 我用简书帖的代码非常的乱呀
  • 阿群1986:闹钟,如果设计成双进程互相唤醒,虽比较流氓,但。。。作为安卓闹钟使用者,“安卓不支持关机状态下播放闹钟音乐并唤醒手机”的这种体验不如塞班时代
  • 94fb3e73f095:少一个类啊,import com.bigkoo.pickerview.TimePickerView;这个类工程里没有
  • c4aa3cd76f5f:假如一个闹钟定的是周一,周二,周五早上八点,是不是先把周一二五持久化存储,然后周一的闹钟响了,再设置周二的。不知道我这个思路对不对?如果用户间隔时间不规律的话应该只能这样做了吧?

本文标题:关于Android中设置闹钟的相对完善的解决方案

本文链接:https://www.haomeiwen.com/subject/igytdttx.html