What to Do if Your Broadcastreceiver is Continuously Repeating Code
Alarms (based on the AlarmManager
class) give you a way to perform time-based operations outside the lifetime of your application. For example, you could use an alarm to initiate a long-running operation, such as starting a service once a day to download a weather forecast.
Alarms have these characteristics:
-
They let you fire Intents at set times and/or intervals.
-
You can use them in conjunction with broadcast receivers to schedule jobs or WorkRequests to perform other operations.
-
They operate outside of your application, so you can use them to trigger events or actions even when your app is not running, and even if the device itself is asleep.
-
They help you minimize your app's resource requirements. You can schedule operations without relying on timers or continuously running services.
Note: For timing operations that are guaranteed to occur during the lifetime of your application, instead consider using the Handler
class in conjunction with Timer
and Thread
. This approach gives Android better control over system resources.
Set an inexact alarm
When an app sets an inexact alarm, the system doesn't promise the delivery of the alarm at an exact point of time in the future. Instead, the system delivers this alarm at a time when it thinks it's most efficient for the device's battery.
Inexact alarms provide some guarantees regarding the timing of alarm delivery, as long as no battery-saving restrictions such as Doze are in effect. The following sections describe these API guarantees.
Deliver an alarm after a specific time
If your app calls set()
, setInexactRepeating()
, or setAndAllowWhileIdle()
, the alarm never goes off before the supplied trigger time.
On Android 12 (API level 31) and higher, the system invokes the alarm within one hour of the supplied trigger time, unless any battery-saving restrictions are in effect such as battery saver or Doze.
Deliver an alarm during a time window
If your app calls setWindow()
, the alarm never goes off before the supplied trigger time. Unless any battery-saving restrictions are in effect, the alarm is delivered within the specified time window, starting from the given trigger time.
If your app targets Android 12 or higher, the system can delay the invocation of an inexact alarm by at least 10 minutes. For this reason, windowLengthMillis
parameter values under 600000
are clipped to 600000
. If a feature in your app requires a higher time precision, use exact alarms instead.
Deliver a repeating alarm at roughly regular intervals
If your app calls setInexactRepeating()
, the system invokes multiple alarms:
- The first alarm goes off within the specified time window, starting from the given trigger time.
- Subsequent alarms usually go off after the specified time window elapses. The time between two consecutive invocations of the alarm can vary.
Set an exact alarm
The system invokes an exact alarm at a precise moment in the future. If your app targets Android 12 or higher, you must declare one of the "Alarms & reminders" permissions; otherwise, a SecurityException
occurs.
Your app can set exact alarms using one of the following methods. These methods are ordered such that the ones closer to the bottom of the list serve more time-critical tasks but demand more system resources.
-
setExact()
-
Invoke an alarm at a nearly precise time in the future, as long as other battery-saving measures aren't in effect.
Use this method to set exact alarms, unless your app's work is time-critical for the user.
-
setExactAndAllowWhileIdle()
-
Invoke an alarm at a nearly precise time in the future, even if battery-saving measures are in effect.
-
setAlarmClock()
-
Invoke an alarm at a precise time in the future. Because these alarms are highly visible to users, the system never adjusts their delivery time. The system identifies these alarms as the most critical ones and leaves low-power modes if necessary to deliver the alarms.
System resource consumption
When the system triggers exact alarms that your app sets, the device consumes a great deal of resources, such as battery life, especially if it's in a power-saving mode. Furthermore, the system cannot easily batch these requests in order to use resources more efficiently.
It's highly recommended that you create an inexact alarm whenever possible. To perform longer work, schedule it using WorkManager
or JobScheduler
from your alarm's BroadcastReceiver
. To perform work while the device is in Doze, create an inexact alarm using setAndAllowWhileIdle()
, and start a job from the alarm.
Declare the appropriate exact alarm permission
If your app targets Android 12 or higher, you must obtain the "Alarms & reminders" special app access. To do so, declare the SCHEDULE_EXACT_ALARM
permission in your app's manifest file, as shown in the following code snippet:
<manifest ...> <uses-permission android:name="android.permission.SCHEDULE_EXACT_ALARM"/> <application ...> ... </application> </manifest>
If your app targets Android 13 (API level 33) or higher, you have the option to declare either the SCHEDULE_EXACT_ALARM
or the USE_EXACT_ALARM
permission.
<manifest ...> <uses-permission android:name="android.permission.USE_EXACT_ALARM"/> <application ...> ... </application> </manifest>
While both the SCHEDULE_EXACT_ALARM
and the USE_EXACT_ALARM
permissions signal the same capabilities, they are granted differently and support different use-cases. Your app should use exact alarms, and declare either SCHEDULE_EXACT_ALARM
or USE_EXACT_ALARM
permission, only if a user-facing function in your app requires precisely-timed actions.
USE_EXACT_ALARM
- Granted automatically
- Cannot be revoked by the user
- Subject to an upcoming Google Play policy
- Limited use cases
SCHEDULE_EXACT_ALARM
- Granted by the user
- Broader set of use cases
- Apps should confirm that the permission has not been revoked
Acceptable use cases for USE_EXACT_ALARM
The following situations are considered acceptable uses of the USE_EXACT_ALARM
permission:
- Your app is an alarm clock app or a timer app.
- Your app is a calendar app that shows notifications for upcoming events.
Using the SCHEDULE_EXACT_ALARM
permission
Unlike USE_EXACT_ALARM
, the SCHEDULE_EXACT_ALARM
permission must be granted by the user. Both the user and the system can revoke the SCHEDULE_EXACT_ALARM
permission.
To check whether the permission is granted to your app, call canScheduleExactAlarms()
before trying to set an exact alarm. When the SCHEDULE_EXACT_ALARM
permission is revoked for your app, your app stops, and all future exact alarms are canceled. This also means that the value returned by canScheduleExactAlarms()
stays valid for the entire lifecycle of your app.
When the SCHEDULE_EXACT_ALARMS
permission is granted to your app, the system sends it the ACTION_SCHEDULE_EXACT_ALARM_PERMISSION_STATE_CHANGED
broadcast. Your app should implement a broadcast receiver that does the following:
- Confirms that your app still has the special app access. To do so, call
canScheduleExactAlarms()
. This check protects your app from the case where the user grants your app the permission, then revokes it almost immediately afterward. - Reschedules any exact alarms that your app needs, based on its current state. This logic should be similar to what your app does when it receives the
ACTION_BOOT_COMPLETED
broadcast.
Ask users to grant the SCHEDULE_EXACT_ALARM
permission
If necessary, you can send users to the Alarms & reminders screen in system settings, as shown in Figure 1. To do so, complete the following steps:
- In your app's UI, explain to the user why your app needs to schedule exact alarms.
- Invoke an intent that includes the
ACTION_REQUEST_SCHEDULE_EXACT_ALARM
intent action.
Set a repeating alarm
Repeating alarms allow the system to notify your app on a recurring schedule.
A poorly-designed alarm can cause battery drain and put a significant load on servers. For this reason, on Android 4.4 (API level 19) and higher, all repeating alarms are inexact alarms.
A repeating alarm has the following characteristics:
- An alarm type. For more discussion, see Choose an alarm type.
- A trigger time. If the trigger time you specify is in the past, the alarm triggers immediately.
- The alarm's interval. For example, once a day, every hour, every 5 minutes, and so on.
- A pending intent that fires when the alarm is triggered. When you set a second alarm that uses the same pending intent, it replaces the original alarm.
To cancel a PendingIntent
, pass FLAG_NO_CREATE
to PendingIntent.getService()
to get an instance of the intent (if it exists), then pass that intent to 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); }
Choose an alarm type
One of the first considerations in using a repeating alarm is what its type should be.
There are two general clock types for alarms: "elapsed real time" and "real time clock" (RTC). Elapsed real time uses the "time since system boot" as a reference, and real time clock uses UTC (wall clock) time. This means that elapsed real time is suited to setting an alarm based on the passage of time (for example, an alarm that fires every 30 seconds) since it isn't affected by time zone/locale. The real time clock type is better suited for alarms that are dependent on current locale.
Both types have a "wakeup" version, which says to wake up the device's CPU if the screen is off. This ensures that the alarm will fire at the scheduled time. This is useful if your app has a time dependency—for example, if it has a limited window to perform a particular operation. If you don't use the wakeup version of your alarm type, then all the repeating alarms will fire when your device is next awake.
If you simply need your alarm to fire at a particular interval (for example, every half hour), use one of the elapsed real time types. In general, this is the better choice.
If you need your alarm to fire at a particular time of day, then choose one of the clock-based real time clock types. Note, however, that this approach can have some drawbacks—the app may not translate well to other locales, and if the user changes the device's time setting, it could cause unexpected behavior in your app. Using a real time clock alarm type also does not scale well, as discussed above. We recommend that you use a "elapsed real time" alarm if you can.
Here is the list of types:
-
ELAPSED_REALTIME
—Fires the pending intent based on the amount of time since the device was booted, but doesn't wake up the device. The elapsed time includes any time during which the device was asleep. -
ELAPSED_REALTIME_WAKEUP
—Wakes up the device and fires the pending intent after the specified length of time has elapsed since device boot. -
RTC
—Fires the pending intent at the specified time but does not wake up the device. -
RTC_WAKEUP
—Wakes up the device to fire the pending intent at the specified time.
Examples of elapsed real time alarms
Here are some examples of using ELAPSED_REALTIME_WAKEUP
.
Wake up the device to fire the alarm in 30 minutes, and every 30 minutes after that:
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);
Wake up the device to fire a one-time (non-repeating) alarm in one minute:
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);
Examples of real time clock alarms
Here are some examples of using RTC_WAKEUP
.
Wake up the device to fire the alarm at approximately 2:00 p.m., and repeat once a day at the same time:
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);
Wake up the device to fire the alarm at precisely 8:30 a.m., and every 20 minutes thereafter:
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);
Decide how precise your alarm needs to be
As described above, choosing the alarm type is often the first step in creating an alarm. A further distinction is how precise you need your alarm to be. For most apps, setInexactRepeating()
is the right choice. When you use this method, Android synchronizes multiple inexact repeating alarms and fires them at the same time. This reduces the drain on the battery.
For the rare app that has rigid time requirements—for example, the alarm needs to fire precisely at 8:30 a.m., and every hour on the hour thereafter—set an exact alarm by calling setRepeating()
. But you should avoid using exact alarms if possible.
With setInexactRepeating()
, you can't specify a custom interval the way you can with setRepeating()
. You have to use one of the interval constants, such as INTERVAL_FIFTEEN_MINUTES
, INTERVAL_DAY
, and so on. See AlarmManager
for the complete list.
Cancel an alarm
Depending on your app, you may want to include the ability to cancel the alarm. To cancel an alarm, call cancel()
on the Alarm Manager, passing in the PendingIntent
you no longer want to fire. For example:
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); }
Start an alarm when the device restarts
By default, all alarms are canceled when a device shuts down. To prevent this from happening, you can design your application to automatically restart a repeating alarm if the user reboots the device. This ensures that the AlarmManager
will continue doing its task without the user needing to manually restart the alarm.
Here are the steps:
- Set the
RECEIVE_BOOT_COMPLETED
permission in your application's manifest. This allows your app to receive theACTION_BOOT_COMPLETED
that is broadcast after the system finishes booting (this only works if the app has already been launched by the user at least once):<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
- Implement a
BroadcastReceiver
to receive the broadcast: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. } } }
- Add the receiver to your app's manifest file with an intent filter that filters on the
ACTION_BOOT_COMPLETED
action:<receiver android:name=".SampleBootReceiver" android:enabled="false"> <intent-filter> <action android:name="android.intent.action.BOOT_COMPLETED"></action> </intent-filter> </receiver>
Notice that in the manifest, the boot receiver is set to
android:enabled="false"
. This means that the receiver will not be called unless the application explicitly enables it. This prevents the boot receiver from being called unnecessarily. You can enable a receiver (for example, if the user sets an alarm) as follows: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);
Once you enable the receiver this way, it will stay enabled, even if the user reboots the device. In other words, programmatically enabling the receiver overrides the manifest setting, even across reboots. The receiver will stay enabled until your app disables it. You can disable a receiver (for example, if the user cancels an alarm) as follows:
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);
Invoke alarms while the device is in Doze mode
Devices that run Android 6.0 (API level 23) support Doze mode, which helps extend device battery life. Alarms do not fire when the device is in Doze mode. Any scheduled alarms are deferred until the device exits Doze. If you need to complete work even when the device is idle, there are several options available:
- Set an exact alarm.
- Use the WorkManager API, which is built to perform background work. You can indicate that the system should expedite your work so that the work finishes as soon as possible. For more information, see Schedule tasks with WorkManager.
Best practices
Every choice you make in designing your repeating alarm can have consequences in how your app uses (or abuses) system resources. For example, imagine a popular app that syncs with a server. If the sync operation is based on clock time and every instance of the app syncs at 11:00 p.m., the load on the server could result in high latency or even "denial of service." Follow these best practices in using alarms:
- Add randomness (jitter) to any network requests that trigger as a result of a repeating alarm:
- Do any local work when the alarm triggers. "Local work" means anything that doesn't hit a server or require the data from the server.
- At the same time, schedule the alarm that contains the network requests to fire at some random period of time.
- Keep your alarm frequency to a minimum.
- Don't wake up the device unnecessarily (this behavior is determined by the alarm type, as described in Choose an alarm type).
- Don't make your alarm's trigger time any more precise than it has to be.
Use
setInexactRepeating()
instead ofsetRepeating()
. When you usesetInexactRepeating()
, Android synchronizes repeating alarms from multiple apps and fires them at the same time. This reduces the total number of times the system must wake the device, thus reducing drain on the battery. As of Android 4.4 (API Level 19), all repeating alarms are inexact. Note that whilesetInexactRepeating()
is an improvement oversetRepeating()
, it can still overwhelm a server if every instance of an app hits the server around the same time. Therefore, for network requests, add some randomness to your alarms, as discussed above. - Avoid basing your alarm on clock time if possible.
Repeating alarms that are based on a precise trigger time don't scale well. Use
ELAPSED_REALTIME
if you can. The different alarm types are described in more detail in the following section.
Source: https://developer.android.com/training/scheduling/alarms
0 Response to "What to Do if Your Broadcastreceiver is Continuously Repeating Code"
Post a Comment