Android: Jak używać AlarmManager

89

Potrzebuję wyzwolić blok kodu po 20 minutach od AlarmManagerustawienia.

Czy ktoś może pokazać mi przykładowy kod dotyczący korzystania z an AlarmManagerAndroida?

Bawiłem się kodem od kilku dni i po prostu nie działa.

Tomek
źródło

Odpowiedzi:

109

„Jakiś przykładowy kod” nie jest takie proste AlarmManager.

Oto fragment przedstawiający konfigurację 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);

W tym przykładzie używam setRepeating(). Jeśli chcesz jednorazowego alarmu, po prostu użyj set(). Pamiętaj, aby podać czas rozpoczęcia alarmu w tej samej podstawie czasu, w jakiej został użyty parametr początkowy do set(). W powyższym przykładzie używam AlarmManager.ELAPSED_REALTIME_WAKEUP, więc moja podstawa czasu to SystemClock.elapsedRealtime().

Oto większy przykładowy projekt pokazujący tę technikę.

CommonsWare
źródło
2
Witam ponownie. Dziękuję za odpowiedź. Jeśli kupię twoją książkę, czy wyjaśnia ona szczegółowo, jak wdrożyć menedżera alarmów?
Tom
7
Książka Advanced Android (wersja 0.9) ma około 9 stron obejmujących AlarmManager, WakeLocks i resztę tego przykładu. Prawdopodobnie rozwinie się to nieco w wersji 1.0, gdy wprowadzę poprawkę, o której wspomniałem w mojej odpowiedzi powyżej. Jeśli masz pytania dotyczące książki lub jej przykładowego kodu, przejdź do groups.google.com/group/cw-android, a z przyjemnością na nie odpowiem.
CommonsWare,
17
Każdy programista Androida powinien mieć subskrypcję książek Marka :) Przynajmniej raz
Bostone
1
@ MarioGalván: Musisz to ustawić, gdy Twoja aplikacja jest uruchamiana po raz pierwszy i po ponownym uruchomieniu.
CommonsWare
Myślę, że powinieneś użyć AlarmManager.RTC_WAKEUP, jeśli chcesz, aby uruchamiał się od razu, a następnie co PERIOD. W twoim kodzie będzie uruchamiany po SystemClock.elapsedRealtime (), a następnie po każdym PERIOD.
Damon Yuan
66

W przykładowym kodzie systemu Android jest kilka dobrych przykładów

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

Te do sprawdzenia to:

  • AlarmController.java
  • OneShotAlarm.java

Po pierwsze, potrzebujesz odbiornika, czegoś, co może nasłuchiwać alarmu, gdy zostanie uruchomiony. Dodaj następujące elementy do pliku AndroidManifest.xml

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

Następnie utwórz następującą klasę

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

Następnie, aby wywołać alarm, użyj następujących elementów (na przykład w swojej głównej aktywności):

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

.


Albo jeszcze lepiej, stwórz klasę, która zajmie się tym wszystkim i używaj tego w ten sposób

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

w ten sposób masz to wszystko w jednym miejscu (nie zapomnij edytować 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();
     }
}
Domyślna
źródło
2
Cześć! Przetestowałem ten kod i działał dobrze (+1). ale próbowałem tego dla wielu alarmów (np. jeden na 10 sekund, a drugi na 15 i tylko ten sencond jest odpalany. Czy robię coś źle, czy to jakiś król problemów? EDYCJA: OK, znalazłem tutaj problem : stackoverflow.com/questions/2844274/…
Nuno Gonçalves
FWIW, użyłbym do tego metody statycznej zamiast konstruktora.
Edward Falk
9

Musisz najpierw stworzyć intencję, którą musisz zaplanować. Następnie uzyskaj pendingIntent tego zamiaru. Możesz zaplanować działania, usługi i transmisje. Aby zaplanować działanie, np. MyActivity:

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

Daj to oczekująceIntent to 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);

Teraz MyActivity zostanie uruchomione po 5 sekundach od uruchomienia aplikacji, niezależnie od zatrzymania aplikacji lub przejścia urządzenia w stan uśpienia (z powodu opcji RTC_WAKEUP). Możesz przeczytać kompletny przykładowy kod Planowanie działań, usług i transmisji #Android

SohailAziz
źródło
+1 świetna odpowiedź, dokładnie to, czego potrzebowałem, działający przykład.
A. Alqadomi,
4

Chciałem skomentować, ale <50 powtórzeń, więc proszę. Przyjazne przypomnienie, że jeśli używasz wersji 5.1 lub nowszej i używasz interwału krótszego niż minuta, dzieje się tak:

Suspiciously short interval 5000 millis; expanding to 60 seconds

Zobacz tutaj .

Kurt
źródło
3

Przykładowy kod, gdy chcesz zadzwonić do usługi z Alarmmanagera:

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

Nie musisz prosić użytkownika o uprawnienia.

Cyfrowe przedsięwzięcie reklamowe
źródło
Bardzo powszechny skrót.
Phantômaxx
0

AlarmManager służy do wyzwalania kodu w określonym czasie.

Aby uruchomić Menedżera alarmów, musisz najpierw pobrać instancję z systemu. Następnie przekaż PendingIntent, który zostanie wykonany w określonym przez Ciebie czasie w przyszłości

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

Podczas korzystania z Menedżera alarmów należy zachować ostrożność. Zwykle menedżer alarmów nie może powtórzyć przed upływem minuty. Również w trybie niskiego zużycia energii czas trwania może wydłużyć się do 15 minut.

Faxriddin Abdullayev
źródło