AlarmManager запускает аварийные сигналы в неподходящее время

Мне удалось все правильно создать службу уведомлений, используемую для запуска уведомления в результате тревоги. К сожалению, настройка будильника с помощью 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 аварийных сигналов, которые должны срабатывать еженедельно.

Даже после прочтения многочисленных ответов на подобные вопросы (я не собираюсь создавать дублирующий вопрос), мне не удалось найти проблему.

Ответы

Ответ 1

Для уровней api ниже 19 вы должны использовать AlarmManager.setRepeating(), и ваши тревоги будут срабатывать точно в указанное время. Ты на уровнях 19 и выше, больше не будет работать. В андроиде произошли изменения, так что все повторяющиеся сигналы тревоги неточны. Поэтому, если вы хотите добиться точной повторяющейся тревоги, вы должны запланировать будильник с помощью AlarmManager.setExact(), а затем, когда будильник срабатывает снова на следующей неделе и так далее каждую неделю.

Ответ 2

Из-за setInexactRepeating. Используйте setRepeating, и он будет обработан в нужное время.

Вместо:

setInexactRepeating 

использовать

setRepeating

setInexactRepeating, совместим с ОС и аккумулятором, он объединяет всю работу, выполняемую при получении сигнала тревоги, и работает один за другим, а поскольку setRepeating мгновенно запускает аварийный сигнал

Также примечание: Аварийные сигналы стираются после перезагрузки телефона, возможно, вам придется внедрить приемник для вещания, чтобы сделать его постоянным. Убедитесь, что вы не выполняете эту рабочую среду, вам нужно реализовать ее в Manifest else, когда ваше приложение не находится в фоновом режиме, вы не получите никаких передач.

Небольшой пример:

Это рабочий код. Он разворачивает процессор каждые 10 минут, пока телефон не выключится.

Добавить в 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);
        }
    }
}