Programar alarmes

Os alarmes (com base na classe AlarmManager ) oferecem uma maneira de realizar operações baseadas em tempo fora da vida útil do seu app. Você pode usar um alarme para iniciar uma operação de longa duração, por exemplo, para iniciar um serviço uma vez por dia para fazer o download de uma previsão do tempo.

Os alarmes têm estas características:

  • Permitem que você dispare intents em horários e/ou intervalos definidos

  • Você pode usá-los com broadcast receivers para programar jobs ou WorkRequests para realizar outras operações.

  • Eles operam fora do seu app, de modo que você pode usá-los para acionar eventos ou ações mesmo quando o app não está em execução e até mesmo se o próprio dispositivo está inativo.

  • Ajudam a minimizar os requisitos de recursos do app. Você pode programar operações sem depender de timers ou serviços em execução contínua.

Definir um alarme não exato

Quando um app define um alarme impreciso, o sistema envia o alarme em algum momento no futuro. Os alarmes inexatos oferecem algumas garantias sobre o tempo de entrega do alarme, respeitando as restrições de economia de bateria, como o Soneca.

Os desenvolvedores podem aproveitar as seguintes garantias de API para personalizar o horário de entrega de alarmes inexatos.

Tocar um alarme depois de um horário específico

Se o app chamar set(), setInexactRepeating() ou setAndAllowWhileIdle(), o alarme nunca vai disparar antes do horário de ativação fornecido.

No Android 12 (nível 31 da API) e versões mais recentes, o sistema invoca o alarme em até uma hora do horário de ativação fornecido, a menos que restrições de economia de bateria estejam em vigor, como a Economia de bateria ou o modo Soneca.

Tocar um alarme durante um período

Se o app chamar setWindow(), o alarme nunca vai disparar antes do horário de acionamento fornecido. A menos que haja restrições de economia de bateria em vigor, o alarme será entregue na janela de tempo especificada, começando no horário de acionamento.

Se o app for direcionado ao Android 12 ou versões mais recentes, o sistema poderá atrasar a invocação de um alarme não exato com janela de tempo em pelo menos 10 minutos. Por esse motivo, os valores de parâmetro windowLengthMillis em 600000 são cortados para 600000.

Tocar um alarme recorrente em intervalos regulares

Se o app chamar setInexactRepeating(), o sistema vai invocar vários alarmes:

  1. O primeiro alarme é acionado dentro do período especificado, começando no horário de acionamento fornecido.
  2. Os alarmes subsequentes geralmente tocam depois que o período especificado termina. O tempo entre duas invocações consecutivas do alarme pode variar.

Definir um alarme exato

O sistema invoca um alarme exato em um momento preciso no futuro.

A maioria dos apps pode agendar tarefas e eventos usando alarmes imprecisos para concluir vários casos de uso comuns. Se a funcionalidade principal do app depender de um alarme com horário preciso, como um app de relógio ou agenda, use um alarme exato.

Casos de uso que podem não exigir alarmes exatos

A lista a seguir mostra fluxos de trabalho comuns que podem não exigir um alarme exato:

Programação de operações de tempo durante o ciclo de vida do app
A classe Handler inclui vários métodos bons para processar operações de tempo, como fazer algum trabalho a cada n segundos, enquanto o app está ativo: postAtTime() e postDelayed(). Essas APIs dependem do tempo de atividade do sistema e não do tempo real.
Trabalho programado em segundo plano, como atualizar o app e fazer upload de registros
O WorkManager oferece uma maneira de programar trabalhos periódicos com condição de tempo. É possível fornecer um intervalo de repetição e flexInterval (mínimo de 15 minutos) para definir o ambiente de execução granular para o trabalho.
Ação especificada pelo usuário que acontece após um horário específico (mesmo se o sistema estiver inativo)
Use um alarme não exato. Chame setAndAllowWhileIdle() especificamente.
Ação especificada pelo usuário que acontece após um horário específico
Use um alarme não exato. Chame set() especificamente.
Ação especificada pelo usuário que pode acontecer em um período especificado
Use um alarme não exato. Chame setWindow() especificamente. Se o app for direcionado ao Android 12 ou mais recente, a menor duração de janela permitida será de 10 minutos.

Como definir um alarme exato

O app pode definir alarmes exatos usando um dos seguintes métodos. Esses métodos são ordenados de forma que os mais próximos da parte de baixo da lista atendam a mais tarefas críticas em relação ao tempo, mas exigem mais recursos do sistema.

setExact()

Invocar um alarme em um horário quase preciso no futuro, desde que outras medidas de economia de bateria não estejam em vigor.

Use esse método para definir alarmes exatos, a menos que o trabalho do app seja essencial para o usuário.

setExactAndAllowWhileIdle()

Ativar um alarme em um horário quase preciso no futuro, mesmo que as medidas de economia de bateria estejam em vigor.

setAlarmClock()

Invoca um alarme em um horário preciso no futuro. Como esses alarmes são altamente visíveis para os usuários, o sistema nunca ajusta o horário de entrega deles. O sistema identifica esses alarmes como os mais críticos e sai dos modos de baixo consumo de energia, se necessário, para entregar os alarmes.

Consumo de recursos do sistema

Quando o sistema aciona alarmes exatos definidos pelo app, o dispositivo consome uma grande quantidade de recursos, como a duração da bateria, especialmente se estiver em um modo de economia de energia. Além disso, o sistema não consegue agrupar facilmente essas solicitações para usar os recursos com mais eficiência.

Recomendamos que você crie um alarme inexato sempre que possível. Para realizar um trabalho mais longo, programe-o usando WorkManager ou JobScheduler do BroadcastReceiver do alarme. Para realizar o trabalho enquanto o dispositivo está no modo Soneca, crie um alarme impreciso usando setAndAllowWhileIdle() e inicie um job com o alarme.

Declarar a permissão de alarme exato adequada

Se o app for direcionado ao Android 12 ou versões mais recentes, será necessário obter o acesso especial "Alarmes e lembretes". Para fazer isso, declare a permissão SCHEDULE_EXACT_ALARM no arquivo de manifesto do app, conforme mostrado no snippet de código a seguir:

<manifest ...>
    <uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/>
    <application ...>
        ...
    </application>
</manifest>

Se o app for destinado ao Android 13 (nível 33 da API) ou versões mais recentes, você poderá declarar a permissão SCHEDULE_EXACT_ALARM ou USE_EXACT_ALARM.

<manifest ...>
    <uses-permission android:name="android.permission.USE_EXACT_ALARM"/>
    <application ...>
        ...
    </application>
</manifest>

Embora as permissões SCHEDULE_EXACT_ALARM e USE_EXACT_ALARM indiquem as mesmas funcionalidades, elas são concedidas de maneiras diferentes e oferecem suporte a casos de uso distintos. O app precisa usar alarmes exatos e declarar a permissão SCHEDULE_EXACT_ALARM ou USE_EXACT_ALARM apenas se uma função voltada para o usuário exigir ações no momento certo.

USE_EXACT_ALARM

SCHEDULE_EXACT_ALARM

  • Concedida pelo usuário
  • Conjunto mais amplo de casos de uso
  • Os apps precisam confirmar que a permissão não foi revogada

A permissão SCHEDULE_EXACT_ALARM não é concedida previamente a novas instalações de apps destinados ao Android 13 (nível 33 da API) e versões mais recentes. Se um usuário transferir dados do app para um dispositivo com o Android 14 usando uma operação de backup e restauração, a permissão SCHEDULE_EXACT_ALARM será negada no novo dispositivo. No entanto, se um app já tiver essa permissão, ela será concedida previamente quando o dispositivo for atualizado para o Android 14.

Observação: se o alarme exato for definido usando um objeto OnAlarmListener, como na API setExact, a permissão SCHEDULE_EXACT_ALARM não será necessária.

Como usar a permissão SCHEDULE_EXACT_ALARM

Ao contrário de USE_EXACT_ALARM, a permissão SCHEDULE_EXACT_ALARM precisa ser concedida pelo usuário. O usuário e o sistema podem revogar a permissão SCHEDULE_EXACT_ALARM.

Para verificar se a permissão foi concedida ao app, chame canScheduleExactAlarms() antes de tentar definir um alarme exato. Quando a permissão SCHEDULE_EXACT_ALARM é revogada para seu app, ele é interrompido e todos os alarmes exatos futuros são cancelados. Isso também significa que o valor retornado por canScheduleExactAlarms() permanece válido durante todo o ciclo de vida do app.

Quando a permissão SCHEDULE_EXACT_ALARMS é concedida ao app, o sistema envia a transmissão ACTION_SCHEDULE_EXACT_ALARM_PERMISSION_STATE_CHANGED. O app precisa implementar um broadcast receiver que faz o seguinte:

  1. Confirma se o app ainda tem o acesso especial. Para fazer isso, chame o método canScheduleExactAlarms(). Essa verificação protege o app contra o caso em que o usuário concede a permissão e a revoga quase imediatamente depois.
  2. Reprograma os alarmes exatos necessários para o app com base no estado atual. Essa lógica precisa ser parecida com a de quando o app recebe a transmissão ACTION_BOOT_COMPLETED.

Pedir aos usuários para conceder a permissão SCHEDULE_EXACT_ALARM

A opção &quot;Autorizar a definição de alarmes e lembretes&quot;
Figura 1. Página de acesso especial aos "Alarmes e lembretes" nas configurações do sistema, onde os usuários podem autorizar o app a definir alarmes exatos.

Se necessário, é possível direcionar os usuários para a tela Alarmes e lembretes nas configurações do sistema, como mostrado na Figura 1. Para isso, siga estas etapas:

  1. Na IU do app, explique ao usuário por que o app precisa programar alarmes exatos.
  2. Invoque uma intent que inclua a ação da intent ACTION_REQUEST_SCHEDULE_EXACT_ALARM.

Definir um alarme recorrente

Os alarmes repetidos permitem que o sistema notifique seu app em uma programação recorrente.

Um alarme mal projetado pode consumir bateria e sobrecarregar os servidores. Por isso, no Android 4.4 (nível 19 da API) e versões mais recentes, todos os alarmes recorrentes são alarmes imprecisos.

Um alarme recorrente tem as seguintes características:

  • Um tipo de alarme. Para saber mais, consulte Escolher um tipo de alarme.

  • Um tempo de acionamento. Se o tempo do acionador especificado estiver no passado, o alarme será acionado imediatamente.

  • O intervalo do alarme. Por exemplo, uma vez por dia, a cada hora ou a cada cinco minutos.

  • Um intent pendente que é acionado quando o alarme é disparado. Quando você define um segundo alarme que usa o mesmo intent pendente, ele substitui o alarme original.

Para cancelar um PendingIntent(), transmita FLAG_NO_CREATE para PendingIntent.getService() para receber uma instância do intent (se houver) e transmita esse intent para AlarmManager.cancel()

Kotlin

val alarmManager =
    context.getSystemService(Context.ALARM_SERVICE) as? AlarmManager
val pendingIntent =
    PendingIntent.getService(context, requestId, intent,
                                PendingIntent.FLAG_NO_CREATE)
if (pendingIntent != null && alarmManager != null) {
  alarmManager.cancel(pendingIntent)
}

Java

AlarmManager alarmManager =
    (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
PendingIntent pendingIntent =
    PendingIntent.getService(context, requestId, intent,
                                PendingIntent.FLAG_NO_CREATE);
if (pendingIntent != null && alarmManager != null) {
  alarmManager.cancel(pendingIntent);
}

Escolher um tipo de alarme

Uma das primeiras considerações sobre o uso de um alarme recorrente é o tipo dele.

Há dois tipos de relógio geral para alarmes: "tempo real decorrido" e "relógio em tempo real" (RTC, na sigla em inglês). O relógio "tempo real decorrido" usa o "tempo desde a inicialização do sistema" como referência, e o "relógio em tempo real" se adequa ao fuso horário local. Isso significa que o tempo real decorrido é adequado para definir um alarme com base na passagem do tempo (por exemplo, um alarme disparado a cada 30 segundos), porque não é afetado por fuso horário ou local. O tipo "relógio em tempo real" é mais adequado para alarmes que dependem da localidade atual.

Os dois tipos têm uma versão de "ativação", que diz para ativar a CPU do dispositivo se a tela estiver desligada. Isso garante que o alarme seja disparado no horário programado. Isso é útil se o app tiver uma dependência de tempo. Por exemplo, se ele tiver uma janela limitada para realizar uma operação específica. Se você não usar a versão de ativação do seu tipo de alarme, todos os alarmes recorrentes serão disparados quando o dispositivo estiver novamente em estado de alerta.

Se você simplesmente precisar que seu alarme dispare em um intervalo específico (por exemplo, a cada meia hora), use um dos tipos de "tempo real decorrido". Em geral, essa é a melhor escolha.

Se você precisar ativar o alarme em um determinado horário do dia, escolha um dos tipos de relógio em tempo real com relógio. No entanto, essa abordagem pode ter algumas desvantagens. O app pode não ser traduzido corretamente para outras localidades, e se o usuário mudar a configuração de hora do dispositivo, isso pode causar um comportamento inesperado no app. O uso de um tipo de alarme de relógio em tempo real também não é escalonável, conforme discutido acima. Recomendamos que você use um alarme de "tempo real decorrido", se puder.

Veja a lista de tipos:

  • ELAPSED_REALTIME: dispara o intent pendente com base no tempo decorrido desde que o dispositivo foi inicializado, mas não ativa o dispositivo. O tempo decorrido inclui qualquer momento em que o dispositivo esteve inativo.

  • ELAPSED_REALTIME_WAKEUP: ativa o dispositivo e dispara o intent pendente após o tempo especificado desde a inicialização do dispositivo.

  • RTC: dispara o intent pendente no horário especificado, mas não ativa o dispositivo.

  • RTC_WAKEUP: ativa o dispositivo para disparar o intent pendente no horário especificado.

Exemplos de alarmes de "tempo real decorrido"

Confira alguns exemplos de como usar ELAPSED_REALTIME_WAKEUP

Ativar o dispositivo para disparar o alarme em 30 minutos e a cada 30 minutos depois disso:

Kotlin

// Hopefully your alarm will have a lower frequency than this!
alarmMgr?.setInexactRepeating(
        AlarmManager.ELAPSED_REALTIME_WAKEUP,
        SystemClock.elapsedRealtime() + AlarmManager.INTERVAL_HALF_HOUR,
        AlarmManager.INTERVAL_HALF_HOUR,
        alarmIntent
)

Java

// Hopefully your alarm will have a lower frequency than this!
alarmMgr.setInexactRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP,
        SystemClock.elapsedRealtime() + AlarmManager.INTERVAL_HALF_HOUR,
        AlarmManager.INTERVAL_HALF_HOUR, alarmIntent);

Ativar o dispositivo para disparar um alarme único (não recorrente) em um minuto:

Kotlin

private var alarmMgr: AlarmManager? = null
private lateinit var alarmIntent: PendingIntent
...
alarmMgr = context.getSystemService(Context.ALARM_SERVICE) as AlarmManager
alarmIntent = Intent(context, AlarmReceiver::class.java).let { intent ->
    PendingIntent.getBroadcast(context, 0, intent, 0)
}

alarmMgr?.set(
        AlarmManager.ELAPSED_REALTIME_WAKEUP,
        SystemClock.elapsedRealtime() + 60 * 1000,
        alarmIntent
)

Java

private AlarmManager alarmMgr;
private PendingIntent alarmIntent;
...
alarmMgr = (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
Intent intent = new Intent(context, AlarmReceiver.class);
alarmIntent = PendingIntent.getBroadcast(context, 0, intent, 0);

alarmMgr.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,
        SystemClock.elapsedRealtime() +
        60 * 1000, alarmIntent);

Exemplos de alarmes de "relógio em tempo real"

Confira alguns exemplos de como usar RTC_WAKEUP.

Ativar o dispositivo para disparar o alarme aproximadamente às 14h e repetir uma vez por dia na mesma hora:

Kotlin

// Set the alarm to start at approximately 2:00 p.m.
val calendar: Calendar = Calendar.getInstance().apply {
    timeInMillis = System.currentTimeMillis()
    set(Calendar.HOUR_OF_DAY, 14)
}

// With setInexactRepeating(), you have to use one of the AlarmManager interval
// constants--in this case, AlarmManager.INTERVAL_DAY.
alarmMgr?.setInexactRepeating(
        AlarmManager.RTC_WAKEUP,
        calendar.timeInMillis,
        AlarmManager.INTERVAL_DAY,
        alarmIntent
)

Java

// Set the alarm to start at approximately 2:00 p.m.
Calendar calendar = Calendar.getInstance();
calendar.setTimeInMillis(System.currentTimeMillis());
calendar.set(Calendar.HOUR_OF_DAY, 14);

// With setInexactRepeating(), you have to use one of the AlarmManager interval
// constants--in this case, AlarmManager.INTERVAL_DAY.
alarmMgr.setInexactRepeating(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(),
        AlarmManager.INTERVAL_DAY, alarmIntent);

Ativar o dispositivo para disparar o alarme exatamente às 8h30 e, a partir daí, a cada 20 minutos:

Kotlin

private var alarmMgr: AlarmManager? = null
private lateinit var alarmIntent: PendingIntent
...
alarmMgr = context.getSystemService(Context.ALARM_SERVICE) as AlarmManager
alarmIntent = Intent(context, AlarmReceiver::class.java).let { intent ->
    PendingIntent.getBroadcast(context, 0, intent, 0)
}

// Set the alarm to start at 8:30 a.m.
val calendar: Calendar = Calendar.getInstance().apply {
    timeInMillis = System.currentTimeMillis()
    set(Calendar.HOUR_OF_DAY, 8)
    set(Calendar.MINUTE, 30)
}

// setRepeating() lets you specify a precise custom interval--in this case,
// 20 minutes.
alarmMgr?.setRepeating(
        AlarmManager.RTC_WAKEUP,
        calendar.timeInMillis,
        1000 * 60 * 20,
        alarmIntent
)

Java

private AlarmManager alarmMgr;
private PendingIntent alarmIntent;
...
alarmMgr = (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
Intent intent = new Intent(context, AlarmReceiver.class);
alarmIntent = PendingIntent.getBroadcast(context, 0, intent, 0);

// Set the alarm to start at 8:30 a.m.
Calendar calendar = Calendar.getInstance();
calendar.setTimeInMillis(System.currentTimeMillis());
calendar.set(Calendar.HOUR_OF_DAY, 8);
calendar.set(Calendar.MINUTE, 30);

// setRepeating() lets you specify a precise custom interval--in this case,
// 20 minutes.
alarmMgr.setRepeating(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(),
        1000 * 60 * 20, alarmIntent);

Decidir a precisão que o alarme precisa ter

Conforme descrito acima, escolher o tipo geralmente é o primeiro passo para criar um alarme. Outra distinção é a precisão do alarme. Para a maioria dos apps, setInexactRepeating() é a escolha certa. Quando você usa esse método, o Android sincroniza vários alarmes recorrentes imprecisos e os aciona ao mesmo tempo. Isso reduz o consumo da bateria.

Evite usar alarmes exatos, se possível. No entanto, para o raro app que tem requisitos de tempo rígidos, é possível definir um alarme exato chamando setRepeating().

Com setInexactRepeating(), não é possível especificar um intervalo personalizado como você faz com setRepeating(). Você precisa usar uma das constantes de intervalo, por exemplo, INTERVAL_FIFTEEN_MINUTES, INTERVAL_DAY, e assim por diante. Consulte AlarmManager para ver a lista completa.

Cancelar um alarme

Dependendo do app, convém incluir a capacidade de cancelar o alarme. Para cancelar um alarme, chame cancel() no Gerenciador de alarmes, informando o PendingIntent que você não quer mais disparar. Exemplo:

Kotlin

// If the alarm has been set, cancel it.
alarmMgr?.cancel(alarmIntent)

Java

// If the alarm has been set, cancel it.
if (alarmMgr!= null) {
    alarmMgr.cancel(alarmIntent);
}

Iniciar um alarme quando o dispositivo é reiniciado

Por padrão, todos os alarmes são cancelados quando um dispositivo é desligado. Para evitar que isso aconteça, você pode projetar seu app para reiniciar automaticamente um alarme recorrente se o usuário reinicializar o dispositivo. Isso garante que o AlarmManager vai continuar fazendo a tarefa sem que o usuário precise reiniciar manualmente o alarme.

Veja as etapas:

  1. Defina a permissão RECEIVE_BOOT_COMPLETED no manifesto do seu app. Isso permite que ele receba a ACTION_BOOT_COMPLETED que é transmitida após a inicialização do sistema. Isso só funcionará se o app já tiver sido iniciado pelo usuário pelo menos uma vez:

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
  2. Implemente um BroadcastReceiver para receber a transmissão:

    Kotlin

    class SampleBootReceiver : BroadcastReceiver() {
    
        override fun onReceive(context: Context, intent: Intent) {
            if (intent.action == "android.intent.action.BOOT_COMPLETED") {
                // Set the alarm here.
            }
        }
    }

    Java

    public class SampleBootReceiver extends BroadcastReceiver {
    
        @Override
        public void onReceive(Context context, Intent intent) {
            if (intent.getAction().equals("android.intent.action.BOOT_COMPLETED")) {
                // Set the alarm here.
            }
        }
    }
  3. Adicione o receptor ao arquivo de manifesto do seu app com um filtro de intent que filtre a ação ACTION_BOOT_COMPLETED:

    <receiver android:name=".SampleBootReceiver"
            android:enabled="false">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED"></action>
        </intent-filter>
    </receiver>

    No manifesto, o receptor de inicialização está definido como android:enabled="false". Isso significa que o receptor só será chamado se o app habilitá-lo explicitamente. Isso impede que o receptor de inicialização seja chamado desnecessariamente. Você pode ativar um receptor (por exemplo, se o usuário definir um alarme) da seguinte maneira:

    Kotlin

    val receiver = ComponentName(context, SampleBootReceiver::class.java)
    
    context.packageManager.setComponentEnabledSetting(
            receiver,
            PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
            PackageManager.DONT_KILL_APP
    )

    Java

    ComponentName receiver = new ComponentName(context, SampleBootReceiver.class);
    PackageManager pm = context.getPackageManager();
    
    pm.setComponentEnabledSetting(receiver,
            PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
            PackageManager.DONT_KILL_APP);

    Depois que você ativa o receptor dessa maneira, ele permanece ativado, mesmo se o usuário reinicializar o dispositivo. Em outras palavras, a ativação do receptor de forma programática substitui a configuração do manifesto, mesmo durante as reinicializações. O receptor vai ficar ativado até que o app o desative. Você pode desativar um receptor (por exemplo, se o usuário cancelar um alarme) da seguinte forma:

    Kotlin

    val receiver = ComponentName(context, SampleBootReceiver::class.java)
    
    context.packageManager.setComponentEnabledSetting(
            receiver,
            PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
            PackageManager.DONT_KILL_APP
    )

    Java

    ComponentName receiver = new ComponentName(context, SampleBootReceiver.class);
    PackageManager pm = context.getPackageManager();
    
    pm.setComponentEnabledSetting(receiver,
            PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
            PackageManager.DONT_KILL_APP);

Invocar alarmes enquanto o dispositivo está no modo Soneca

Dispositivos com o Android 6.0 (nível 23 da API) oferecem suporte ao modo Soneca, que ajuda a aumentar a vida útil da bateria do dispositivo. Os alarmes não são acionados quando o dispositivo está no modo Soneca. Todos os alarmes programados são adiados até que o dispositivo saia desse modo. Se você precisa concluir o trabalho mesmo quando o dispositivo está inativo, há várias opções disponíveis:

  • Definir um alarme exato.

  • Use a API WorkManager, que foi criada para executar trabalho em segundo plano. Você pode indicar que o sistema deve acelerar seu trabalho para que ele seja concluído o mais rápido possível. Para mais informações, consulte Agendar tarefas com o WorkManager

Práticas recomendadas

Cada escolha que você faz ao criar um alarme recorrente pode ter consequências na forma como o app usa os (ou abusa dos) recursos do sistema. Por exemplo, imagine um app muito usado que é sincronizado com um servidor. Se a operação de sincronização for baseada no horário do relógio e todas as instâncias do app forem sincronizadas às 23h, a carga no servidor poderá resultar em alta latência ou até mesmo "negação de serviço". Siga estas práticas recomendadas para o uso de alarmes:

  • Adicione aleatoriedade (instabilidade) a todas as solicitações de rede acionadas como resultado de um alarme recorrente:

    • Faça qualquer trabalho local quando o alarme for acionado. "Trabalho local" significa qualquer coisa que não chegue a um servidor ou que exija os dados do servidor.

    • Ao mesmo tempo, programe o alarme que contém as solicitações de rede para disparar em um período aleatório.

  • Mantenha a frequência do alarme em um nível mínimo.

  • Não acione o dispositivo desnecessariamente (esse comportamento é determinado pelo tipo de alarme, conforme descrito em Escolher um tipo de alarme).

  • Não faça com que o horário de acionamento do alarme seja mais preciso do que o necessário.

    Use setInexactRepeating() em vez de setRepeating(). Quando você usa setInexactRepeating(), o Android sincroniza alarmes recorrentes de vários apps e os aciona ao mesmo tempo. Isso diminui o número total de vezes que o sistema precisa ativar o dispositivo, reduzindo, assim, o consumo da bateria. A partir do Android 4.4 (API de nível 19), todos os alarmes recorrentes são alarmes imprecisos. Observe que, embora setInexactRepeating() seja uma melhoria em relação a setRepeating(), ele ainda poderá sobrecarregar um servidor se todas as instâncias de um app chegarem ao servidor ao mesmo tempo. Portanto, para solicitações de rede, adicione alguma aleatoriedade aos seus alarmes, como discutido anteriormente.

  • Evite basear seu alarme na hora do relógio, se possível.

    Alarmes recorrentes baseados em um gatilho preciso não funcionam bem. Use ELAPSED_REALTIME, se for possível. Os diferentes tipos de alarme são descritos em detalhes na seção a seguir.