首页 文章

AlarmManager在错误的时间触发警报

提问于
浏览
15

我管理一切都可以创建一个通知服务,用于通过警报触发通知 . 不幸的是,使用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个警报,每周应该发射一次 .

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

2 回答

  • 14

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

  • 25

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

    代替:

    setInexactRepeating
    

    使用

    setRepeating
    

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

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

    一个小例子:

    This is working code. It wakes CPU every 10 minutes until the phone turns off.

    添加到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);
         }
     }
    

    Set Alarm from Service:

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

    If you want set alarm repeating at phone boot time:

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

相关问题