17

我管理一切正常,以创建一个通知服务,用于在警报时触发通知。不幸的是,使用 AlarmManager 设置警报无法正常工作。它会在几分钟后触发(不完全是几个小时,这表明存在时区问题)。循环周期为 1 周,因此我使用常量 INTERVAL_DAY 并将其乘以 7。为了确保一个 PendingIntent 不会替换另一个,我将 dayOfWeek 作为第二个参数传递给 PendingIntent.getService()。我通过记录来检查警报触发时间的正确性:

Log.d(TAG, "next alarm " + df.format(cal.getTime()));

真的没有办法列出所有设置的警报 - 至少是我自己的应用程序中的那些?我相信这是追踪错误的唯一方法。

我的代码:

cal.setTimeInMillis(System.currentTimeMillis());
cal.add(Calendar.DATE, 1);
cal.set(Calendar.HOUR_OF_DAY, hour);
cal.set(Calendar.MINUTE, minute);
Log.d(TAG, "next alarm " + df.format(cal.getTime()));
Intent showNotificationIntent = new Intent(context, NotificationService.class);
dayOfWeek = cal.get(Calendar.DAY_OF_WEEK);
alarmIntent = PendingIntent.getService(context, dayOfWeek, showNotificationIntent, 0);
getAlarmManager(context).setInexactRepeating(AlarmManager.RTC_WAKEUP, cal.getTimeInMillis(),
    INTERVAL_WEEK, alarmIntent);

我想提供每天都有一个闹钟,但在不同的时间,可以由用户设置。所以我最多使用 7 个警报,应该每周触发一次。

即使在阅读了类似问题的众多答案(我不打算创建重复的问题)之后,我也没有设法找到问题所在。

4

2 回答 2

27

对于低于 19 的 api 级别,您应该使用AlarmManager.setRepeating()并且您的警报将在指定时间准确触发。你在 api 级别 19 及以上将不再工作。android 发生了变化,因此所有重复的警报都是不准确的。因此,如果您想实现精确的重复警报,您应该安排警报,AlarmManager.setExact()然后在警报触发时再次执行下周,依此类推。

于 2014-01-30T16:04:09.767 回答
14

因为 setInexactRepeating。使用 setRepeating 它将在正确的时间进行处理。

代替:

setInexactRepeating 

利用

setRepeating

setInexactRepeating,对操作系统和电池友好,它将所有要在警报接收上完成的工作批处理在一起,并一一完成,而 setRepeating 会立即触发警报

另请注意:一旦手机重新启动,警报就会消失,您可能必须实现启动广播接收器以使其持久化。确保您不执行该运行时,您需要在 Manifest 中实现它,否则当您的应用程序不在后台时,您将不会收到任何广播。

一个小例子:

这是工作代码。它每 10 分钟唤醒一次 CPU,直到手机关机。

添加到 Manifest.xml:

...
<uses-permission android:name="android.permission.WAKE_LOCK"></uses-permission>
...
<receiver  android:process=":remote" android:name="Alarm"></receiver>
...

代码:

    package YourPackage;
    import android.app.AlarmManager;
    import android.app.PendingIntent;
    import android.content.BroadcastReceiver;
    import android.content.Context;
    import android.content.Intent;
    import android.os.PowerManager;
    import android.widget.Toast;

    public class Alarm extends BroadcastReceiver 
    {    
         @Override
         public void onReceive(Context context, Intent intent) 
         {   
             PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
             PowerManager.WakeLock wl = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "");
             wl.acquire();

             // Put here YOUR code.
             Toast.makeText(context, "Alarm !!!!!!!!!!", Toast.LENGTH_LONG).show(); // For example

             wl.release();
         }

     public void SetAlarm(Context context)
     {
         AlarmManager am=(AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
         Intent i = new Intent(context, Alarm.class);
         PendingIntent pi = PendingIntent.getBroadcast(context, 0, i, 0);
         am.setRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis(), 1000 * 60 * 10, pi); // Millisec * Second * Minute
     }

     public void CancelAlarm(Context context)
     {
         Intent intent = new Intent(context, Alarm.class);
         PendingIntent sender = PendingIntent.getBroadcast(context, 0, intent, 0);
         AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
         alarmManager.cancel(sender);
     }
 }

从服务设置警报:

package YourPackage;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.os.IBinder;

public class YourService extends Service
{
    Alarm alarm = new Alarm();
    public void onCreate()
    {
        super.onCreate();       
    }

    public void onStart(Context context,Intent intent, int startId)
    {
        alarm.SetAlarm(context);
    }

    @Override
    public IBinder onBind(Intent intent) 
    {
        return null;
    }
}

如果您想在手机启动时设置重复闹钟:

向 Manifest.xml 添加权限:

<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"></uses-permission>
...
<receiver android:name=".AutoStart">
    <intent-filter>
        <action android:name="android.intent.action.BOOT_COMPLETED"></action>
    </intent-filter>
</receiver>
...

并创建新类:

package YourPackage;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;

public class AutoStart extends BroadcastReceiver
{   
    Alarm alarm = new Alarm();
    @Override
    public void onReceive(Context context, Intent intent)
    {   
        if (intent.getAction().equals("android.intent.action.BOOT_COMPLETED"))
        {
            alarm.SetAlarm(context);
        }
    }
}
于 2014-01-30T15:55:55.590 回答