Android:使用AlarmManager获取所有设置的PendingIntents


71

我正在设置这样的警报:

alarmManager.set(AlarmManager.RTC_WAKEUP, alarmTime, pendingEvent);

我有兴趣删除以前设置的所有警报,然后将其清除。

有什么方法可以让我执行此操作,或者获取当前设置的所有警报,以便可以手动删除它们?


8
:这家伙演示了如何列出当前报警 stackoverflow.com/questions/6522792/...
洛特

Answers:


65

您需要创建待处理的意图,然后将其取消

 AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);

    Intent updateServiceIntent = new Intent(context, MyPendingIntentService.class);
    PendingIntent pendingUpdateIntent = PendingIntent.getService(context, 0, updateServiceIntent, 0);

    // Cancel alarms
    try {
        alarmManager.cancel(pendingUpdateIntent);
    } catch (Exception e) {
        Log.e(TAG, "AlarmManager update was not canceled. " + e.toString());
    }

43
无需参考。AlarmManager.cancel取消所有它们的文档:删除所有具有匹配Intent的警报。任何警报,其意图与该警报相匹配(由filterEquals(Intent)定义)的任何类型的警报都将被取消。
Alex Volovoy

5
如何将对它们的引用保存在数据库中,然后如何对其进行引用?
Arjit 2011年

6
如果我的意图具有某些附加值或捆绑包,该怎么办...我需要创建具有相同附加值的相同意图才能取消该意图吗?
FinalDark 2013年

2
还需要执行未完成的UpdateIntent.cancelt()使其完全工作
Soham

11
匹配Intents时不考虑@FinalDark附加功能,因此Intent创建它时的附加功能没有什么区别。您可以使用Intent无额外功能取消警报。
David Wasser 2014年

83

您不必继续引用它。只需定义一个新的PendingIntent,就像您在创建它时定义的一样即可。

例如:

如果我创建了一个要由AlarmManager触发的PendingIntent,如下所示:

Intent alarmIntent = new Intent(getApplicationContext(), AlarmBroadcastReceiver.class);
alarmIntent.setData(Uri.parse("custom://" + alarm.ID));
alarmIntent.setAction(String.valueOf(alarm.ID));
AlarmManager alarmManager = (AlarmManager) getSystemService(ALARM_SERVICE);

PendingIntent displayIntent = PendingIntent.getBroadcast(getApplicationContext(), 0, alarmIntent, 0);

alarmManager.set(AlarmManager.RTC_WAKEUP, alarmDateTime, displayIntent);

然后可以在其他代码(甚至是其他活动)的某处取消此操作:

Intent alarmIntent = new Intent(getApplicationContext(), AlarmBroadcastReceiver.class);
alarmIntent.setData(Uri.parse("custom://" + alarm.ID));
alarmIntent.setAction(String.valueOf(alarm.ID));
AlarmManager alarmManager = (AlarmManager) getSystemService(ALARM_SERVICE);

PendingIntent displayIntent = PendingIntent.getBroadcast(getApplicationContext(), 0, alarmIntent, 0);

alarmManager.cancel(displayIntent);

此处重要的事情是使用完全相同的数据和操作以及其他条件以及此处所述的http://developer.android.com/reference/android/app/AlarmManager.html#cancel%28android.app设置PendingIntent .PendingIntent%29


6
+1表示alarmIntent.setData(Uri.parse(“ custom://” + alarm.ID)); alarmIntent.setAction(String.valueOf(alarm.ID));
dharam 2013年

1
@ gilsaints88:您能解释一下,alarm.ID的功能是什么,如何在我的两个活动中都声明呢?
mas_bejo 2014年

2
@mas_bejo setData(...)上的alarm.ID是警报的唯一性,以便我们可以在使用setAction(...)时清除具有相同alarm.ID的任何先前设置的警报,以便从通知中传递意图是通过onNewIntent()返回到Activity时的正确值。我在设置警报时使用setData和setAction,并且从通知发出的Activity收到警报时,我通过覆盖Activity的onNewIntent来获得意图。我得到的意图是正确的,因为之前有setAction(...)。我希望这有帮助。
gilsaints88 2014年

嗨,通过setData和SetAction通知在我的OPPO手机中被取消,但在我的MOTO手机中没有被取消。你能建议一下吗?
尼拉夫·帕特尔

@NiravPatel请问您的OPPO是哪个Android版本,您的MOTO是哪个Android版本?谢谢
gilsaints88 '16

20

要取消警报,您需要重新创建相同的PendingIntent并传递相同的请求代码。

因此,我创建了一个AlarmUtils来帮助我将所有请求代码保存在首选项中,以便将来在需要时将其取消。您需要使用以下类并调用以下方法:

  • addAlarm 添加新警报并传递请求代码。
  • cancelAlarm要删除警报,您需要重新创建相同的
    Intent并传递相同的请求代码。
  • hasAlarm 要验证是否添加了该警报,您需要重新创建相同的Intent并传递相同的请求代码。
  • cancelAllAlarms 删除所有设置的警报。

我的AlarmUtils

public class AlarmUtils {

    private static final String sTagAlarms = ":alarms";

    public static void addAlarm(Context context, Intent intent, int notificationId, Calendar calendar) {

        AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
        PendingIntent pendingIntent = PendingIntent.getBroadcast(context, notificationId, intent, PendingIntent.FLAG_CANCEL_CURRENT);

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            alarmManager.setExactAndAllowWhileIdle(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), pendingIntent);
        } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            alarmManager.setExact(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), pendingIntent);
        } else {
            alarmManager.set(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), pendingIntent);
        }

        saveAlarmId(context, notificationId);
    }

    public static void cancelAlarm(Context context, Intent intent, int notificationId) {
        AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
        PendingIntent pendingIntent = PendingIntent.getBroadcast(context, notificationId, intent, PendingIntent.FLAG_CANCEL_CURRENT);
        alarmManager.cancel(pendingIntent);
        pendingIntent.cancel();

        removeAlarmId(context, notificationId);
    }

    public static void cancelAllAlarms(Context context, Intent intent) {
        for (int idAlarm : getAlarmIds(context)) {
            cancelAlarm(context, intent, idAlarm);
        }
    }

    public static boolean hasAlarm(Context context, Intent intent, int notificationId) {
        return PendingIntent.getBroadcast(context, notificationId, intent, PendingIntent.FLAG_NO_CREATE) != null;
    }

    private static void saveAlarmId(Context context, int id) {
        List<Integer> idsAlarms = getAlarmIds(context);

        if (idsAlarms.contains(id)) {
            return;
        }

        idsAlarms.add(id);

        saveIdsInPreferences(context, idsAlarms);
    }

    private static void removeAlarmId(Context context, int id) {
        List<Integer> idsAlarms = getAlarmIds(context);

        for (int i = 0; i < idsAlarms.size(); i++) {
            if (idsAlarms.get(i) == id)
                idsAlarms.remove(i);
        }

        saveIdsInPreferences(context, idsAlarms);
    }

    private static List<Integer> getAlarmIds(Context context) {
        List<Integer> ids = new ArrayList<>();
        try {
            SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
            JSONArray jsonArray2 = new JSONArray(prefs.getString(context.getPackageName() + sTagAlarms, "[]"));

            for (int i = 0; i < jsonArray2.length(); i++) {
                ids.add(jsonArray2.getInt(i));
            }

        } catch (Exception e) {
            e.printStackTrace();
        }

        return ids;
    }

    private static void saveIdsInPreferences(Context context, List<Integer> lstIds) {
        JSONArray jsonArray = new JSONArray();
        for (Integer idAlarm : lstIds) {
            jsonArray.put(idAlarm);
        }

        SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
        SharedPreferences.Editor editor = prefs.edit();
        editor.putString(context.getPackageName() + sTagAlarms, jsonArray.toString());

        editor.apply();
    }
}

可能值得removeAlarmId公开,这样当通知到达时人们可以删除ID。
Baggers

您可以使用cancelAlarm()它代替,因为您需要创建未决的
Intent

有趣的是,所以即使警报已通过,它仍然排入队列?
Baggers

1
@VadimKotov您可能已经在使用androidx,因此您需要在实现gradle依赖项中添加实现'androidx.preference:preference:1.1.0'才能使用未弃用的androidx PrefenerceManager。

1
支持有史以来最桃子的答案,感谢您节省我的时间
ucMedia

0

如果将PendingIntent ID保存在SharedPreference或db中,则在Android重新启动强制停止程序包时,将得到错误的值

您可以将以前的所有PendingIntent保存到另一个新的PendingIntent中,然后再取消所有

例如

    void createSomeAlarmAndSave() {
        PendingIntent pendingIntent = PendingIntent.getBroadcast(this, 0, new Intent(this, MyReceiver.class).setAction(ACTION_XXX), 0);
        getSystemService(AlarmManager.class).setInexactRepeating(AlarmManager.RTC_WAKEUP, 1000 * 60 * 15, 1000 * 60 * 15, pendingIntent);

        PendingIntent pendingIntent1 = PendingIntent.getBroadcast(this, 1, new Intent(this, MyReceiver.class).setAction(ACTION_XXX), 0);
        getSystemService(AlarmManager.class).setInexactRepeating(AlarmManager.RTC_WAKEUP, 1000 * 60 * 20, 1000 * 60 * 20, pendingIntent1);

        PendingIntent pendingIntent2 = PendingIntent.getBroadcast(this, 2, new Intent(this, MyReceiver.class).setAction(ACTION_XXX), 0);
        getSystemService(AlarmManager.class).setInexactRepeating(AlarmManager.RTC_WAKEUP, 1000 * 60 * 25, 1000 * 60 * 25, pendingIntent2);

        //save all previous PendingIntent to another new PendingIntent
        PendingIntent.getBroadcast(this, 0, new Intent(this, MyReceiver.class).putExtra("previous", new PendingIntent[]{pendingIntent, pendingIntent1, pendingIntent2}), PendingIntent.FLAG_UPDATE_CURRENT);
    }

取消所有先前的PendingIntent

    void cancelAllPreviousAlarm() {
        //acquire the dedicated PendingIntent
        PendingIntent pendingIntentAllPrevious = PendingIntent.getBroadcast(this, 0, new Intent(this, MyReceiver.class), PendingIntent.FLAG_NO_CREATE);
        if (pendingIntentAllPrevious != null) {
            try {
                pendingIntentAllPrevious.send(this, 0, null, new PendingIntent.OnFinished() {
                    @Override
                    public void onSendFinished(PendingIntent pendingIntent, Intent intent, int resultCode, String resultData, Bundle resultExtras) {
                        for (Parcelable parcelable : intent.getParcelableArrayExtra("previous")) {
                            //alarm will cancel when the corresponding PendingIntent cancel
                            ((PendingIntent) parcelable).cancel();
                        }
                    }
                }, null);
                pendingIntentAllPrevious.cancel();
            } catch (PendingIntent.CanceledException e) {
                e.printStackTrace();
            }
        }
    }
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.