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