Android: como usar o AlarmManager

89

Preciso acionar um bloco de código após 20 minutos a partir da AlarmManagerdefinição.

Alguém pode me mostrar um exemplo de código sobre como usar um AlarmManagerِ Android?

Eu tenho brincado com alguns códigos por alguns dias e simplesmente não funciona.

Tom
fonte

Respostas:

109

"Algum código de amostra" não é tão fácil quando se trata de AlarmManager.

Aqui está um snippet que mostra a configuração de AlarmManager:

AlarmManager mgr=(AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
Intent i=new Intent(context, OnAlarmReceiver.class);
PendingIntent pi=PendingIntent.getBroadcast(context, 0, i, 0);

mgr.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime(), PERIOD, pi);

Neste exemplo, estou usando setRepeating(). Se você quiser um alarme instantâneo, basta usar set(). Certifique-se de dar o tempo para o alarme iniciar na mesma base de tempo que você usou no parâmetro inicial para set(). No meu exemplo acima, estou usando AlarmManager.ELAPSED_REALTIME_WAKEUP, então minha base de tempo é SystemClock.elapsedRealtime().

Aqui está um projeto de amostra maior mostrando essa técnica.

CommonsWare
fonte
2
Olá de novo. Obrigado pela resposta. Se eu comprar o seu livro, ele explica como implementar um gerenciador de alarmes em todos os detalhes?
Tom
7
O livro Advanced Android (versão 0.9) tem cerca de 9 páginas cobrindo AlarmManager, WakeLocks e o resto desse exemplo. Isso provavelmente se expandirá um pouco na versão 1.0, conforme eu fizer a correção que mencionei na minha resposta acima. E se você tiver dúvidas sobre o livro ou seu código de amostra, vá para groups.google.com/group/cw-android e terei prazer em respondê-las.
CommonsWare
17
Qualquer desenvolvedor de Android deve ter uma assinatura dos livros de Mark :) Pelo menos uma vez
Bostone
1
@ MarioGalván: Você precisa configurá-lo quando seu aplicativo for executado pela primeira vez e na reinicialização.
CommonsWare
Acho que você deve usar AlarmManager.RTC_WAKEUP se quiser que ele seja acionado imediatamente e a cada PERÍODO. Em seu código, ele será acionado após SystemClock.elapsedRealtime () e a cada PERÍODO.
Damon Yuan
66

Existem alguns bons exemplos no código de amostra do Android

. \ android-sdk \ samples \ android-10 \ ApiDemos \ src \ com \ example \ android \ apis \ app

Os que você deve verificar são:

  • AlarmController.java
  • OneShotAlarm.java

Em primeiro lugar, você precisa de um receptor, algo que possa ouvir o seu alarme quando for disparado. Adicione o seguinte ao seu arquivo AndroidManifest.xml

<receiver android:name=".MyAlarmReceiver" />

Em seguida, crie a seguinte classe

public class MyAlarmReceiver extends BroadcastReceiver { 
     @Override
     public void onReceive(Context context, Intent intent) {
         Toast.makeText(context, "Alarm went off", Toast.LENGTH_SHORT).show();
     }
}

Então, para acionar um alarme, use o seguinte (por exemplo, em sua atividade principal):

AlarmManager alarmMgr = (AlarmManager)getSystemService(Context.ALARM_SERVICE);
Intent intent = new Intent(this, MyAlarmReceiver.class);
PendingIntent pendingIntent = PendingIntent.getBroadcast(this, 0, intent, 0);
Calendar time = Calendar.getInstance();
time.setTimeInMillis(System.currentTimeMillis());
time.add(Calendar.SECOND, 30);
alarmMgr.set(AlarmManager.RTC_WAKEUP, time.getTimeInMillis(), pendingIntent);

.


Ou, melhor ainda, crie uma classe que trate de tudo e use-a assim

Bundle bundle = new Bundle();
// add extras here..
MyAlarm alarm = new MyAlarm(this, bundle, 30);

desta forma, você tem tudo em um só lugar (não se esqueça de editar o AndroidManifest.xml)

public class MyAlarm extends BroadcastReceiver {
    private final String REMINDER_BUNDLE = "MyReminderBundle"; 

    // this constructor is called by the alarm manager.
    public MyAlarm(){ }

    // you can use this constructor to create the alarm. 
    //  Just pass in the main activity as the context, 
    //  any extras you'd like to get later when triggered 
    //  and the timeout
     public MyAlarm(Context context, Bundle extras, int timeoutInSeconds){
         AlarmManager alarmMgr = 
             (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
         Intent intent = new Intent(context, MyAlarm.class);
         intent.putExtra(REMINDER_BUNDLE, extras);
         PendingIntent pendingIntent =
             PendingIntent.getBroadcast(context, 0, intent, 
             PendingIntent.FLAG_UPDATE_CURRENT);
         Calendar time = Calendar.getInstance();
         time.setTimeInMillis(System.currentTimeMillis());
         time.add(Calendar.SECOND, timeoutInSeconds);
         alarmMgr.set(AlarmManager.RTC_WAKEUP, time.getTimeInMillis(),
                      pendingIntent);
     }

      @Override
     public void onReceive(Context context, Intent intent) {
         // here you can get the extras you passed in when creating the alarm
         //intent.getBundleExtra(REMINDER_BUNDLE));

         Toast.makeText(context, "Alarm went off", Toast.LENGTH_SHORT).show();
     }
}
Padrão
fonte
2
Olá! Testei este código e funcionou bem (+1). mas tentei fazer isso para vários alarmes (como um por 10 segundos e outro por 15, e apenas o sencond é disparado. Estou fazendo algo errado ou é algum tipo de problema? EDITAR: Ok, encontrei o problema aqui : stackoverflow.com/questions/2844274/…
Nuno Gonçalves
FWIW, eu usaria um método estático em vez de um construtor para isso.
Edward Falk
9

O que você precisa fazer é primeiro criar a intenção que você precisa agendar. Em seguida, obtenha o pendingIntent dessa intenção. Você pode agendar atividades, serviços e transmissões. Para agendar uma atividade, por exemplo, MyActivity:

  Intent i = new Intent(getApplicationContext(), MyActivity.class);
  PendingIntent pi = PendingIntent.getActivity(getApplicationContext(),3333,i,
  PendingIntent.FLAG_CANCEL_CURRENT);

Dê este pendenteIntent para alarmManager:

  //getting current time and add 5 seconds in it
  Calendar cal = Calendar.getInstance();
  cal.add(Calendar.SECOND, 5);
  //registering our pending intent with alarmmanager
  AlarmManager am = (AlarmManager) getSystemService(ALARM_SERVICE);
  am.set(AlarmManager.RTC_WAKEUP,cal.getTimeInMillis(), pi);

Agora, MyActivity será iniciado após 5 segundos do lançamento do aplicativo, não importa se você interrompeu o aplicativo ou o dispositivo entrou no estado de suspensão (devido à opção RTC_WAKEUP). Você pode ler o código de exemplo completo Agendando atividades, serviços e transmissões #Android

SohailAziz
fonte
1 ótima resposta, exatamente o que eu precisava, um exemplo de 'conjunto' de trabalho.
A.Alqadomi
4

Eu queria comentar, mas <50 repetições, então aqui vai. Lembrete amigável de que se você estiver executando em 5.1 ou superior e usar um intervalo de menos de um minuto, isso acontece:

Suspiciously short interval 5000 millis; expanding to 60 seconds

Veja aqui .

Kurt
fonte
3

Alguns exemplos de código quando você deseja chamar um serviço do Alarmmanager:

PendingIntent pi;
AlarmManager mgr;
mgr = (AlarmManager)ctx.getSystemService(Context.ALARM_SERVICE);
Intent i = new Intent(DataCollectionActivity.this, HUJIDataCollectionService.class);    
pi = PendingIntent.getService(DataCollectionActivity.this, 0, i, 0);
mgr.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime() , 1000, pi);

Você não tem que pedir permissão ao usuário.

The Digital Ad Venture
fonte
Uma abreviatura muito comum.
Phantômaxx de
0

Um AlarmManager é usado para acionar algum código em um momento específico.

Para iniciar um Alarm Manager, você precisa primeiro obter a instância do Sistema. Em seguida, passe o PendingIntent que seria executado em um momento futuro que você especificar

AlarmManager manager = (AlarmManager) getSystemService(Context.ALARM_SERVICE);

Intent alarmIntent = new Intent(context, MyAlarmReceiver.class);
PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0, alarmIntent, 0);
int interval = 8000; //repeat interval
manager.setInexactRepeating(AlarmManager.RTC_WAKEUP, System.currentTimeMillis(), interval, pendingIntent);

Você precisa ter cuidado ao usar o Gerenciador de alarmes. Normalmente, um gerenciador de alarme não pode repetir antes de um minuto. Também no modo de baixa energia, a duração pode aumentar para até 15 minutos.

Faxriddin Abdullayev
fonte