AlarmManager在错误的时间触发警报

我管理一切都可以创建一个通知服务,用于通过警报触发通知。 不幸的是,使用AlarmManager设置警报不能正常工作。 它会在几分钟后发射(不完全是几小时,这表明存在时区问题)。 循环周期为1周,因此我使用常数INTERVAL_DAY并将其乘以7.为了确保一个PendingIntent不替换另一个,我将dayOfWeek作为第二个parameter passing给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个警报,每周应该发射一次。

即使在阅读了类似问题的大量答案之后(我不打算创建一个重复的问题),我也没有设法find问题。

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

因为setInexactRepeating。 使用setRepeating,它将在合适的时间处理。

代替:

 setInexactRepeating 

使用

 setRepeating 

setInexactRepeating,操作系统和电池友好,它将所有要完成的工作批处理接收并逐个工作,同时setRepeating立即触发警报

另请注意:重新启动电话后,警报会被擦除,您可能必须实施启动广播接收器才能使其持久化。 确保你没有那个运行时,你需要在Manifest中实现它,当你的应用程序不在后台你不会收到任何广播。

一个小例子:

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

添加到Manifest.xml:

 ...  ...  ... 

码:

  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的权限:

  ...      ... 

并创建新类:

 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); } } }