AlarmManager
) memberi Anda cara untuk melakukan
operasi berbasis waktu di luar masa aktif aplikasi Anda.
Misalnya, Anda dapat menggunakan alarm untuk memulai operasi yang berjalan lama, seperti
memulai layanan sekali dalam satu hari untuk mendownload perkiraan cuaca.
Alarm memiliki karakteristik berikut:
Alarm memungkinkan Anda memicu Intent pada waktu/interval yang ditentukan.
Anda dapat menggunakannya bersama dengan penerima siaran untuk menjadwalkan tugas atau WorkRequest untuk melakukan operasi lainnya.
Alarm beroperasi di luar aplikasi Anda, sehingga dapat digunakan untuk memicu peristiwa atau tindakan bahkan saat aplikasi Anda sedang tidak berjalan, dan meskipun perangkat itu sendiri dalam mode tidur.
Alarm membantu meminimalkan kebutuhan resource aplikasi Anda. Anda dapat menjadwalkan operasi tanpa mengandalkan timer atau layanan yang berjalan terus-menerus.
Menyetel alarm yang tidak tepat
Saat aplikasi menyetel alarm inexact, sistem akan mengirimkan alarm pada suatu waktu di masa mendatang. Alarm tidak tepat memberikan beberapa jaminan terkait waktu pengiriman alarm sambil mematuhi batasan penghematan baterai seperti Istirahatkan.
Developer dapat memanfaatkan jaminan API berikut untuk menyesuaikan pengaturan waktu pengiriman alarm yang tidak tepat.
Membunyikan alarm setelah waktu tertentu
Jika aplikasi Anda memanggil set()
,
setInexactRepeating()
,
atau setAndAllowWhileIdle()
,
alarm tidak akan berbunyi sebelum waktu pemicu yang diberikan.
Di Android 12 (level API 31) dan yang lebih tinggi, sistem akan memanggil alarm dalam waktu satu jam dari waktu pemicu yang diberikan, kecuali jika ada pembatasan penghematan baterai yang berlaku seperti penghemat baterai atau Istirahatkan.
Menyampaikan alarm selama periode waktu
Jika aplikasi Anda memanggil setWindow()
, alarm tidak akan pernah berbunyi sebelum waktu pemicu
yang diberikan. Kecuali jika ada pembatasan penghematan baterai, alarm akan dikirimkan dalam jangka waktu yang ditentukan, mulai dari waktu pemicu yang diberikan.
Jika aplikasi Anda menargetkan Android 12 atau yang lebih tinggi, sistem dapat menunda
pemanggilan alarm tidak tepat berjangka waktu setidaknya selama 10 menit. Oleh karena itu, nilai parameter windowLengthMillis
di 600000
dipangkas menjadi 600000
.
Menyampaikan alarm berulang pada interval yang kurang lebih teratur
Jika aplikasi Anda memanggil setInexactRepeating()
,
sistem akan memanggil beberapa alarm:
- Alarm pertama berbunyi dalam jangka waktu yang ditentukan, mulai dari waktu pemicu yang diberikan.
- Alarm berikutnya biasanya berbunyi setelah jangka waktu yang ditentukan berakhir. Waktu antara dua pemanggilan alarm berturut-turut dapat bervariasi.
Menyetel alarm yang tepat
Sistem memanggil alarm yang tepat pada waktu yang tepat di masa mendatang.
Sebagian besar aplikasi dapat menjadwalkan tugas dan acara menggunakan alarm yang tidak tepat untuk menyelesaikan beberapa kasus penggunaan umum. Jika fungsi inti aplikasi Anda bergantung pada alarm yang waktunya tepat—seperti untuk aplikasi jam alarm atau aplikasi kalender—maka Anda boleh menggunakan alarm yang tepat.
Kasus penggunaan yang mungkin tidak memerlukan alarm yang tepat
Daftar berikut menunjukkan alur kerja umum yang mungkin tidak memerlukan alarm yang tepat:
- Menjadwalkan operasi pengaturan waktu selama masa aktif aplikasi Anda
- Class
Handler
mencakup beberapa metode yang baik untuk menangani operasi pemilihan waktu, seperti melakukan beberapa tugas setiap n detik, saat aplikasi Anda aktif:postAtTime()
danpostDelayed()
. Perhatikan bahwa API ini mengandalkan waktu aktif sistem dan bukan waktu real-time. - Pekerjaan latar belakang terjadwal, seperti mengupdate aplikasi dan mengupload log
WorkManager
menyediakan cara untuk menjadwalkan pekerjaan berkala yang terikat waktu. Anda dapat memberikan interval pengulangan danflexInterval
(minimum 15 menit) untuk menentukan runtime terperinci bagi tugas.- Tindakan yang ditentukan pengguna yang akan terjadi setelah waktu tertentu (bahkan jika sistem dalam keadaan tidak ada aktivitas)
- Menggunakan alarm yang tidak tepat. Secara khusus, panggil
setAndAllowWhileIdle()
. - Tindakan yang ditentukan pengguna yang akan terjadi setelah waktu tertentu
- Menggunakan alarm yang tidak tepat. Secara khusus, panggil
set()
. - Tindakan yang ditentukan pengguna yang dapat terjadi dalam periode waktu tertentu
- Menggunakan alarm yang tidak tepat. Secara khusus, panggil
setWindow()
. Perhatikan bahwa, jika aplikasi Anda menargetkan Android 12 atau yang lebih tinggi, durasi jendela terkecil yang diizinkan adalah 10 menit.
Cara menyetel alarm yang tepat
Aplikasi Anda dapat menyetel alarm tepat menggunakan salah satu metode berikut. Metode ini diurutkan sedemikian rupa sehingga metode yang lebih dekat ke bagian bawah daftar melayani tugas yang lebih kritis waktu, tetapi memerlukan lebih banyak resource sistem.
setExact()
Memanggil alarm pada waktu yang hampir tepat di masa mendatang, selama tindakan penghematan baterai lainnya tidak berlaku.
Gunakan metode ini untuk menyetel alarm tepat, kecuali jika pekerjaan aplikasi Anda sangat penting bagi pengguna.
setExactAndAllowWhileIdle()
Memanggil alarm pada waktu yang hampir tepat di masa mendatang, meskipun langkah-langkah penghematan baterai sedang diterapkan.
setAlarmClock()
Memanggil alarm pada waktu yang tepat di masa mendatang. Karena alarm ini sangat terlihat oleh pengguna, sistem tidak pernah menyesuaikan waktu pengirimannya. Sistem mengidentifikasi alarm ini sebagai yang paling penting dan keluar dari mode daya rendah jika perlu untuk menyampaikan alarm.
Penggunaan resource sistem
Saat sistem memicu alarm yang tepat yang ditetapkan aplikasi Anda, perangkat akan menggunakan banyak resource, seperti masa pakai baterai, terutama jika dalam mode hemat daya. Selain itu, sistem tidak dapat dengan mudah mengelompokkan permintaan ini untuk menggunakan resource secara lebih efisien.
Sebaiknya buat alarm tidak tepat setiap kali
memungkinkan. Untuk melakukan pekerjaan yang lebih lama, jadwalkan menggunakan
WorkManager
atau
JobScheduler
dari
BroadcastReceiver
alarm Anda. Untuk melakukan pekerjaan saat
perangkat berada dalam mode Istirahatkan, buat alarm yang tidak tepat menggunakan
setAndAllowWhileIdle()
,
dan mulai tugas dari alarm.
Mendeklarasikan izin alarm yang tepat yang sesuai
Jika aplikasi Anda menargetkan Android 12 atau yang lebih tinggi, Anda harus mendapatkan akses aplikasi khusus "Alarm & pengingat". Untuk melakukannya, deklarasikan izin
SCHEDULE_EXACT_ALARM
di file manifes aplikasi Anda, seperti yang ditunjukkan dalam cuplikan kode berikut:
<manifest ...> <uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/> <application ...> ... </application> </manifest>
Jika aplikasi Anda menargetkan Android 13 (level API 33) atau yang lebih tinggi, Anda memiliki opsi untuk
mendeklarasikan izin SCHEDULE_EXACT_ALARM
atau USE_EXACT_ALARM
.
<manifest ...> <uses-permission android:name="android.permission.USE_EXACT_ALARM"/> <application ...> ... </application> </manifest>
Meskipun izin SCHEDULE_EXACT_ALARM
dan USE_EXACT_ALARM
menandakan kemampuan yang sama, keduanya diberikan secara berbeda dan mendukung kasus penggunaan yang berbeda. Aplikasi Anda harus menggunakan alarm yang tepat, dan mendeklarasikan izin SCHEDULE_EXACT_ALARM
atau USE_EXACT_ALARM
, hanya jika fungsi yang dilihat pengguna di aplikasi Anda memerlukan tindakan tepat waktu.
USE_EXACT_ALARM
- Diberikan secara otomatis
- Tidak dapat dibatalkan oleh pengguna
- Tunduk pada kebijakan Google Play mendatang
- Kasus penggunaan terbatas
SCHEDULE_EXACT_ALARM
- Diberikan oleh pengguna
- Kumpulan kasus penggunaan yang lebih luas
- Aplikasi harus mengonfirmasi bahwa izin belum dicabut
Izin SCHEDULE_EXACT_ALARM
tidak diberikan kepada penginstalan baru aplikasi yang menargetkan Android 13 (level API 33) dan yang lebih tinggi. Jika pengguna mentransfer data aplikasi ke perangkat yang menjalankan Android 14 melalui operasi pencadangan dan pemulihan, izin SCHEDULE_EXACT_ALARM
akan ditolak di perangkat baru. Namun, jika
aplikasi yang ada sudah memiliki izin ini, izin tersebut akan diberikan sebelumnya saat
perangkat diupgrade ke Android 14.
Catatan: Jika alarm yang tepat disetel menggunakan objek
OnAlarmListener
, seperti dengan
setExact
API, izin SCHEDULE_EXACT_ALARM
tidak diperlukan.
Menggunakan izin SCHEDULE_EXACT_ALARM
Berbeda dengan USE_EXACT_ALARM
, izin SCHEDULE_EXACT_ALARM
harus
diberikan oleh pengguna. Pengguna dan sistem dapat mencabut izin SCHEDULE_EXACT_ALARM
.
Untuk memeriksa apakah izin diberikan ke aplikasi Anda, panggil
canScheduleExactAlarms()
sebelum mencoba menyetel alarm yang tepat. Jika izin SCHEDULE_EXACT_ALARM
dicabut untuk aplikasi Anda, aplikasi Anda akan berhenti, dan semua alarm tepat di masa mendatang akan dibatalkan. Artinya juga bahwa nilai yang ditampilkan oleh
canScheduleExactAlarms()
tetap valid selama seluruh siklus proses aplikasi Anda.
Jika izin SCHEDULE_EXACT_ALARMS
diberikan ke aplikasi Anda, sistem akan mengirimkan siaran ACTION_SCHEDULE_EXACT_ALARM_PERMISSION_STATE_CHANGED
ke aplikasi tersebut. Aplikasi Anda harus menerapkan penerima
siaran yang melakukan
hal berikut:
- Mengonfirmasi bahwa aplikasi Anda masih memiliki akses aplikasi khusus. Untuk melakukannya,
panggil
canScheduleExactAlarms()
Pemeriksaan ini melindungi aplikasi Anda dari kasus saat pengguna memberikan izin ke aplikasi Anda, lalu mencabutnya hampir segera setelahnya. - Menjadwalkan ulang alarm tepat apa pun yang diperlukan aplikasi Anda, berdasarkan statusnya saat ini.
Logika ini harus mirip dengan yang dilakukan aplikasi Anda saat menerima
siaran
ACTION_BOOT_COMPLETED
.
Meminta pengguna untuk memberikan izin SCHEDULE_EXACT_ALARM
Jika perlu, Anda dapat mengarahkan pengguna ke layar Alarm & pengingat di setelan sistem, seperti yang ditunjukkan pada Gambar 1. Caranya, selesaikan langkah-langkah berikut:
- Di UI aplikasi, jelaskan kepada pengguna alasan aplikasi Anda perlu menjadwalkan alarm yang tepat.
- Panggil intent yang menyertakan
tindakan intent
ACTION_REQUEST_SCHEDULE_EXACT_ALARM
.
Menyetel alarm berulang
Alarm berulang memungkinkan sistem memberi tahu aplikasi Anda sesuai jadwal berulang.
Alarm yang tidak didesain dengan baik dapat menghabiskan baterai dan memberikan beban berat pada server. Oleh karena itu, di Android 4.4 (level API 19) dan yang lebih tinggi, semua alarm berulang adalah alarm tidak pasti.
Alarm berulang memiliki karakteristik berikut:
Jenis alarm. Untuk pembahasan lebih lanjut, buka Memilih jenis alarm.
Waktu pemicu. Jika waktu pemicu yang Anda tentukan adalah di masa lalu, alarm akan langsung terpicu.
Interval alarm. Misalnya, sekali sehari, setiap jam, atau setiap 5 menit.
Intent tertunda yang terpicu saat alarm dipicu. Saat Anda menyetel alarm kedua yang menggunakan intent tertunda yang sama, alarm tersebut akan menggantikan alarm yang asli.
Untuk membatalkan PendingIntent()
, teruskan
FLAG_NO_CREATE
ke PendingIntent.getService()
untuk mendapatkan instance intent (jika ada), lalu teruskan intent tersebut ke
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); }
Memilih jenis alarm
Salah satu pertimbangan awal dalam menggunakan alarm berulang adalah jenisnya.
Ada dua jenis jam umum untuk alarm: "elapsed real time" dan "real time clock" (RTC). Elapsed real time menggunakan "time since system boot" sebagai referensi, dan real time clock menggunakan waktu UTC (jam dinding). Artinya, elapsed real time cocok untuk menyetel alarm berdasarkan waktu yang berlalu (misalnya, alarm yang terpicu setiap 30 detik) karena jenis alarm ini tidak terpengaruh oleh zona waktu atau lokal. Jenis real time clock lebih cocok untuk alarm yang bergantung pada lokal saat ini.
Kedua jenis tersebut memiliki versi "wakeup", yang memberi tahu agar CPU perangkat diaktifkan jika layar nonaktif. Ini akan memastikan bahwa alarm dipicu pada waktu yang dijadwalkan. Hal ini berguna jika aplikasi Anda memiliki dependensi waktu. Misalnya, jika aplikasi memiliki jendela terbatas untuk melakukan operasi tertentu. Jika Anda tidak menggunakan versi wakeup dari jenis alarm Anda, maka semua alarm berulang akan terpicu saat perangkat aktif di waktu berikutnya.
Jika Anda hanya memerlukan agar alarm terpicu pada interval tertentu (misalnya, setiap setengah jam), gunakan salah satu jenis elapsed real time. Secara umum, ini adalah pilihan yang lebih baik.
Jika Anda perlu agar alarm terpicu pada waktu tertentu dalam sehari, pilih salah satu jenis real time clock yang berbasis jam. Namun, perlu diperhatikan bahwa pendekatan ini dapat memiliki beberapa kekurangan. Aplikasi mungkin tidak diterjemahkan dengan baik ke lokalitas lain, dan jika pengguna mengubah setelan waktu perangkat, hal ini dapat menyebabkan perilaku yang tidak terduga di aplikasi Anda. Penggunaan jenis alarm jam real-time juga tidak dapat diskalakan dengan baik, seperti yang dibahas di atas. Sebaiknya gunakan alarm "elapsed real time" jika memungkinkan.
Berikut daftar jenis alarm:
ELAPSED_REALTIME
: Memicu intent tertunda berdasarkan waktu yang berlalu sejak perangkat di-booting, tetapi tidak mengaktifkan perangkat. Waktu berlalu mencakup kapan saja saat perangkat sedang tidur.ELAPSED_REALTIME_WAKEUP
: Mengaktifkan perangkat dan memicu intent tertunda setelah jangka waktu yang ditentukan telah berlalu sejak perangkat di-booting.RTC
: Memicu intent tertunda pada waktu yang ditentukan, tetapi tidak mengaktifkan perangkat.RTC_WAKEUP
: Mengaktifkan perangkat untuk memicu intent tertunda pada waktu yang ditentukan.
Contoh alarm elapsed real time
Berikut beberapa contoh penggunaan ELAPSED_REALTIME_WAKEUP
Mengaktifkan perangkat untuk memicu alarm dalam 30 menit, dan setiap 30 menit setelahnya:
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);
Mengaktifkan perangkat untuk memicu alarm satu kali (tidak berulang) dalam satu menit:
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);
Contoh alarm real time clock
Berikut beberapa contoh penggunaan
RTC_WAKEUP
.
Mengaktifkan perangkat untuk memicu alarm kira-kira pada pukul 14.00, kemudian mengulanginya sekali sehari di waktu yang sama:
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);
Mengaktifkan perangkat untuk memicu alarm tepat pada pukul 08.30, dan setiap 20 menit setelahnya:
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);
Menentukan seberapa tepat alarm Anda seharusnya
Seperti yang dijelaskan sebelumnya, memilih jenis alarm sering kali merupakan langkah pertama dalam
pembuatan alarm. Perbedaan lebih lanjut adalah seberapa tepat alarm Anda seharusnya. Untuk sebagian besar aplikasi,
setInexactRepeating()
adalah pilihan yang tepat. Saat Anda menggunakan metode ini, Android menyinkronkan beberapa alarm berulang tidak pasti dan memicunya
di waktu yang bersamaan. Ini akan mengurangi pemborosan baterai.
Jika memungkinkan, hindari penggunaan alarm yang tepat. Namun, untuk aplikasi langka yang memiliki persyaratan waktu yang kaku, Anda dapat menyetel alarm persis dengan memanggil
setRepeating()
.
Dengan setInexactRepeating()
, Anda tidak dapat menentukan interval kustom seperti yang dapat Anda lakukan dengan setRepeating()
.
Anda harus menggunakan salah satu konstanta interval, seperti
INTERVAL_FIFTEEN_MINUTES
,
INTERVAL_DAY
,
dan sebagainya. Lihat AlarmManager
untuk melihat daftar lengkapnya.
Membatalkan alarm
Bergantung pada aplikasi, Anda mungkin ingin menyertakan kemampuan membatalkan alarm.
Untuk membatalkan alarm, panggil cancel()
di Pengelola Alarm, yang meneruskan
PendingIntent
yang tidak ingin Anda
picu. Contoh:
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); }
Memulai alarm saat perangkat dimulai ulang
Secara default, semua alarm dibatalkan setelah perangkat nonaktif.
Untuk mencegah agar hal ini tidak terjadi, Anda dapat mendesain aplikasi
agar secara otomatis memulai ulang alarm berulang jika pengguna melakukan reboot pada perangkat. Tindakan ini memastikan AlarmManager
akan terus melakukan tugasnya agar pengguna tidak perlu memulai ulang alarm secara manual.
Berikut langkah-langkahnya:
Tetapkan izin
RECEIVE_BOOT_COMPLETED
dalam manifes aplikasi Anda. Hal ini memungkinkan aplikasi Anda menerimaACTION_BOOT_COMPLETED
yang disiarkan setelah sistem selesai melakukan booting (ini hanya berfungsi jika aplikasi telah diluncurkan oleh pengguna minimal satu kali):<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
Terapkan
BroadcastReceiver
untuk menerima siaran: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. } } }
Tambahkan penerima ke file manifes aplikasi Anda dengan filter intent yang memfilter tindakan
ACTION_BOOT_COMPLETED
:<receiver android:name=".SampleBootReceiver" android:enabled="false"> <intent-filter> <action android:name="android.intent.action.BOOT_COMPLETED"></action> </intent-filter> </receiver>
Perhatikan bahwa di manifes, penerima booting disetel ke
android:enabled="false"
. Artinya, penerima tidak akan dipanggil kecuali aplikasi secara eksplisit mengaktifkannya. Tindakan ini mencegah penerima booting dipanggil jika tidak diperlukan. Anda dapat mengaktifkan penerima (misalnya, jika pengguna menyetel alarm) seperti berikut: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);
Setelah Anda mengaktifkan penerima dengan cara ini, penerima akan terus diaktifkan, meskipun pengguna melakukan reboot pada perangkat. Dengan kata lain, pengaktifan penerima melalui program akan menggantikan setelan manifes, bahkan saat perangkat di-reboot. Penerima akan tetap diaktifkan sampai aplikasi Anda menonaktifkannya. Anda dapat menonaktifkan penerima (misalnya, jika pengguna membatalkan alarm) seperti berikut:
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);
Memanggil alarm saat perangkat dalam mode Istirahatkan
Perangkat yang menjalankan Android 6.0 (API level 23) mendukung mode Istirahatkan, yang membantu memperpanjang masa pakai baterai perangkat. Alarm tidak akan berbunyi saat perangkat berada dalam mode Istirahatkan Semua alarm terjadwal akan ditunda hingga perangkat keluar dari mode Istirahatkan. Jika Anda perlu menyelesaikan pekerjaan meskipun perangkat sedang menganggur, ada beberapa opsi yang tersedia:
Menyetel alarm yang tepat.
Gunakan WorkManager API, yang dibuat untuk melakukan pekerjaan latar belakang. Anda dapat menunjukkan bahwa sistem harus mempercepat pekerjaan Anda sehingga pekerjaan selesai sesegera mungkin. Untuk mengetahui informasi selengkapnya, lihat Menjadwalkan tugas dengan WorkManager.
Praktik terbaik
Setiap pilihan yang Anda buat dalam mendesain alarm berulang dapat memiliki konsekuensi terhadap cara aplikasi Anda menggunakan (atau menyalahgunakan) resource sistem. Misalnya, bayangkan suatu aplikasi populer yang disinkronkan dengan server. Jika operasi sinkronisasi didasarkan pada waktu jam dan setiap instance aplikasi disinkronkan pada pukul 23.00, beban terhadap server dapat menyebabkan latensi yang tinggi atau bahkan "penolakan layanan". Ikuti praktik terbaik penggunaan alarm berikut:
Tambahkan pengacakan (jitter) untuk semua permintaan jaringan yang terpicu akibat alarm berulang:
Lakukan semua pekerjaan lokal saat alarm terpicu. "Pekerjaan lokal" berarti semua pekerjaan yang tidak terhubung ke server atau memerlukan data dari server.
Pada saat yang bersamaan, jadwalkan alarm yang berisi permintaan jaringan agar dipicu pada jangka waktu yang acak.
Pertahankan agar frekuensi alarm Anda tetap minimum.
Jangan aktifkan perangkat jika tidak perlu (perilaku ini ditentukan oleh jenis alarm, sebagaimana yang dijelaskan di Memilih jenis alarm).
Jangan membuat waktu pemicu alarm yang lebih tepat dari yang diperlukan.
Gunakan
setInexactRepeating()
, bukansetRepeating()
. Saat Anda menggunakansetInexactRepeating()
, Android menyinkronkan alarm berulang dari beberapa aplikasi dan memicunya di saat yang bersamaan. Tindakan ini akan mengurangi jumlah total waktu yang digunakan sistem untuk mengaktifkan perangkat, sehingga mengurangi pemborosan baterai. Untuk Android 4.4 (API Level 19), semua alarm berulang adalah alarm tidak pasti. Perhatikan bahwa meskipunsetInexactRepeating()
lebih baik daripadasetRepeating()
, kode tersebut masih dapat membebani server jika setiap instance aplikasi mengakses server di saat yang hampir bersamaan. Oleh karena itu, untuk permintaan jaringan, tambahkan beberapa pengacakan ke alarm Anda, seperti yang dibahas sebelumnya.Jika memungkinkan, hindari mendasarkan alarm pada waktu jam.
Alarm berulang yang didasarkan pada waktu pemicu yang tepat tidak akan diskalakan dengan baik. Gunakan
ELAPSED_REALTIME
jika Anda bisa. Berbagai jenis alarm dijelaskan secara lebih mendetail di bagian berikut.