无法检查AlarmManager是否已设置警报

12

我正在使用这个答案检查AlarmManager是否已经设置了闹钟。

以下是我的代码片段。

boolean alarmUp = (PendingIntent.getBroadcast(MainActivity.this, 0,
    new Intent(MainActivity.this, AlarmReceiver.class), PendingIntent.FLAG_NO_CREATE) != null);
if (alarmUp) {
    // alarm is set; do some stuff
}

Intent alarmIntent = new Intent(MainActivity.this, AlarmReceiver.class);
final PendingIntent pendingIntent = PendingIntent.getBroadcast(MainActivity.this, 0, alarmIntent, PendingIntent.FLAG_UPDATE_CURRENT);

AlarmManager manager = (AlarmManager) getSystemService(Context.ALARM_SERVICE);
manager.setInexactRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis(), 10 * 1000, pendingIntent);
然而,alarmUp 一直被设置为true。也就是说,无论我是否设置闹钟,在每次重新启动应用程序时,它都告诉我 alarmUp 是真的(我通过制作一个 Toast 来检查它)。请帮助我找出问题所在。

1
我发现检查PendingIntent是否存在的技术并不可靠。 - CommonsWare
请您能否详细说明一下,或者提供一个链接来说明如何做到这一点? - Kevin Richards
我指的是你问题中的代码。我发现FLAG_NO_CREATE的技巧并不可靠。 - CommonsWare
好的!你知道其他实现我想做的事情的方法吗? - Kevin Richards
3
不行,抱歉。 AlarmManager 需要一个更好的 API。 - CommonsWare
2个回答

17

为了确保此检查有效,您需要绝对确定在设置闹钟时只有PendingIntent存在。有两件事可以做来确保如此:

1)在测试代码时,请确保卸载应用程序并在测试之前重新安装应用程序。卸载应用程序将删除应用程序可能创建但仍处于挂起状态的任何PendingIntent

2)取消闹钟时,请确保同时取消PendingIntent。您可以使用

Intent alarmIntent = new Intent(MainActivity.this, AlarmReceiver.class);
final PendingIntent pendingIntent = 
          PendingIntent.getBroadcast(MainActivity.this, 0, alarmIntent,
          PendingIntent.FLAG_NO_CREATE);
if (pendingIntent != null) {
    pendingIntent.cancel();
}

0

自从 API 21 开始,你可以使用

public AlarmManager.AlarmClockInfo getNextAlarmClock ()

http://developer.android.com/reference/android/app/AlarmManager.html#getNextAlarmClock()


现在,您正在尝试使用的代码是:

(PendingIntent.getBroadcast(MainActivity.this, 0,
    new Intent(MainActivity.this, AlarmReceiver.class), PendingIntent.FLAG_NO_CREATE) != null);

基本上,您正在请求一个名为AlarmReceiver的先前和现有意图。但是他的AlarmReceiver指的是您自己的BroadcastReceiver。

正如您在之前发布的答案中看到的那样:

boolean alarmUp = (PendingIntent.getBroadcast(context, 0, 
    new Intent("com.my.package.MY_UNIQUE_ACTION"), 
    PendingIntent.FLAG_NO_CREATE) != null);

他们使用“MY_UNIQUE_ACTION”来查看意图是否存在。

此外,在这个网站上,您可以看到使用它的教程:

http://justcallmebrian.com/2010/04/27/using-alarmmanager-to-schedule-activities-on-android/


除非您能够访问android系统的AlarmReceiver并查看意图是否存在,否则您将无法请求“通用”的定时闹钟。这是您想要做的吗?如果是这种情况,您确定在没有任何预定闹钟的情况下,android系统的AlarmReceiver意图不会被系统创建吗? 似乎我们无法像我们想要的那样控制这些组件。


一个工作示例: Manifest.xml:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.blablabla.testa" >
<uses-permission android:name="android.permission.WAKE_LOCK" />

<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/AppTheme" >
    <activity
        android:name=".MainActivity"
        android:label="@string/app_name" >
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>

        <receiver
            android:name=".MySuperReceiver"
            android:label="MySuperReceiverName" />

    </activity>
</application>

MySuperReceiver.java

package com.blablabla.testa;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

import java.util.Calendar;

public class MySuperReceiver extends BroadcastReceiver {
    public static final String TAG = MySuperReceiver.class.getSimpleName();
    public static final String ACTION_ALARM_RECEIVER = "ACTION_ALARM_RECEIVER";
    private Calendar c = Calendar.getInstance();

    @Override
    public void onReceive(Context context, Intent intent) {
        if (intent != null)
            if (ACTION_ALARM_RECEIVER.equals(intent.getAction())) {
                Log.d(TAG, new Exception().getStackTrace()[0].getMethodName() + " " + c.getTime());
                //do something here
            }
    }
}

而 MainActivity.java:

package com.blablabla.testa;

import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import android.support.v7.app.ActionBarActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;


public class MainActivity extends ActionBarActivity {

    public static final String TAG = "TEST APP:";

    AlarmManager alarmManager;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        GetAlarmService();
        CreateAlarm();
        CheckForAlarm();
        CancelAlarms();
        CheckForAlarm();
    }

    private void GetAlarmService()
    {
        alarmManager = (AlarmManager) getApplicationContext().getSystemService(Context.ALARM_SERVICE);
        Log.d(TAG, " GET Alarm Service ! ");
    }
    private void CreateAlarm()
    {
        long aroundInterval = 1*60*1000; 

        Intent intent = new Intent(getApplicationContext(), MySuperReceiver.class);
        intent.setAction(MySuperReceiver.ACTION_ALARM_RECEIVER);//my custom string action name

        PendingIntent pendingIntent = PendingIntent.getBroadcast(getApplicationContext(), 1001, intent, PendingIntent.FLAG_CANCEL_CURRENT);//used unique ID as 1001
        alarmManager.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis(), aroundInterval, pendingIntent);//first start will start asap

        Log.d(TAG, " CREATE Alarm ! ");
    }

    private void CancelAlarms()
    {
        Intent intent = new Intent(getApplicationContext(), MySuperReceiver.class);//the same as up
        intent.setAction(MySuperReceiver.ACTION_ALARM_RECEIVER);//the same as up
        PendingIntent pendingIntent = PendingIntent.getBroadcast(getApplicationContext(), 1001, intent, PendingIntent.FLAG_CANCEL_CURRENT);//the same as up
        alarmManager.cancel(pendingIntent);//important
        pendingIntent.cancel();//important

        Log.d(TAG, " Cancel Alarm ! ");
    }

    private void CheckForAlarm()
    {
        //checking if alarm is working with pendingIntent #3
        Intent intent = new Intent(getApplicationContext()  , MySuperReceiver.class);//the same as up
        intent.setAction(MySuperReceiver.ACTION_ALARM_RECEIVER);//the same as up
        boolean isWorking = (PendingIntent.getBroadcast(getApplicationContext() , 1001, intent, PendingIntent.FLAG_NO_CREATE) != null);//just changed the flag
        Log.d("TAG: TEST APP:  ", "alarm is " + (isWorking ? "" : "not") + " working...");

    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();

        //noinspection SimplifiableIfStatement
        if (id == R.id.action_settings) {
            return true;
        }

        return super.onOptionsItemSelected(item);
    }
}

有没有办法让它在 API 版本低于 21 的设备上运行?我想知道的只是,我的应用程序是否已经设置了闹钟。 - Kevin Richards
如果您使用自己的BroadcastReceiver设置了一个闹钟,那么您应该能够读取该寄存器。这里有另一个链接,提供另一个示例:http://blog.mokrzycki.io/2015/01/working-example-of-setting-alarm-with-repeating-stuff-checking-if-alarm-was-set-with-pendingintent/ - mayo
我已经按照你链接中提到的第二点做了同样的事情。而这正是我在问题中提到的不起作用的地方。 - Kevin Richards
你在清单文件中添加了这些行吗?如果你改变了接收器的名称,并检查第一次是否也会在getBroadcast上得到null呢? - mayo
嗯,当你的alarmUp为真时,在取消意图后会怎样?(我开始编码了 :)) - mayo
显示剩余2条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接