Home | History | Annotate | Download | only in scheduling
      1 page.title=Scheduling Repeating Alarms
      2 parent.title=Using Wake Locks
      3 parent.link=index.html
      4 
      5 trainingnavtop=true
      6 
      7 @jd:body
      8 
      9 <div id="tb-wrapper">
     10 <div id="tb">
     11 
     12 <!-- table of contents -->
     13 <h2>This lesson teaches you to</h2>
     14 <ol>
     15   <li><a href="#set">Set a Repeating Alarm</a></li>
     16   <li><a href="#cancel">Cancel an Alarm</a></li>
     17   <li><a href="#boot">Start an Alarm When the Device Boots</a></li>
     18 </ol>
     19 
     20 <h2>Try it out</h2>
     21 
     22 <div class="download-box">
     23   <a href="{@docRoot}shareables/training/Scheduler.zip"
     24 class="button">Download the sample</a>
     25  <p class="filename">Scheduler.zip</p>
     26 </div>
     27 
     28 </div>
     29 </div>
     30 
     31 <p>Alarms (based on the {@link android.app.AlarmManager} class) give you a way to perform
     32 time-based operations outside the lifetime of your application.
     33 For example, you could use an alarm to initiate a long-running operation, such
     34 as starting a service once a day to download a weather forecast.</p>
     35 
     36 <p>Alarms have these characteristics:</p>
     37 <ul>
     38 
     39 <li>They let you fire Intents at set times and/or intervals.</li>
     40 
     41 <li>You can use them in conjunction with broadcast receivers to start services and perform
     42 other operations.</li>
     43 
     44 <li>They operate outside of your application, so you can use them to trigger events or
     45 actions even when your app is not running, and even if the device itself is asleep.</li>
     46 
     47 <li>They help you to minimize your app's resource requirements. You can schedule operations
     48 without relying on timers or continuously running background services.</li>
     49 
     50 </ul>
     51 
     52 <p class="note"><strong>Note:</strong> For timing operations that are guaranteed to occur
     53 <em>during</em> the lifetime of your application,
     54 instead consider using the {@link android.os.Handler} class in conjunction with
     55 {@link java.util.Timer} and {@link java.lang.Thread}. This approach gives Android better
     56 control over system resources.</p>
     57 
     58 <h2 id="set">Set a Repeating Alarm</h2>
     59 
     60 <p>As described above, repeating alarms are a good choice for scheduling regular events or
     61 data lookups. A repeating alarm has the following characteristics:</p>
     62 
     63 <ul>
     64 <li>A alarm type. For more discussion, see <a href="#type">Choose an alarm type</a>.</li>
     65 <li>A trigger time. If the trigger time you specify is in the past, the alarm triggers
     66 immediately.</li>
     67 <li>The alarm's interval. For example, once a day, every hour, every 5 seconds, and so on.</li>
     68 <li>A pending intent that fires when the alarm is triggered. When you set a second alarm
     69 that uses the same pending intent, it replaces the original alarm.</li>
     70 </ul>
     71 
     72 <p>Every choice you make in designing your repeating alarm can have consequences in how your
     73 app uses (or abuses) system resources. Even a carefully managed alarm can have a major impact
     74 on battery life. Follow these guidelines as you design your app:</p>
     75 
     76 <ul>
     77 <li>Keep your alarm frequency to a minimum.</li>
     78 <li>Don't wake up the device unnecessarily (this behavior is determined by the alarm type,
     79 as described in <a href="#type">Choose an alarm type</a>).</li>
     80 <li>Don't make your alarm's trigger time any more precise than it has to be:
     81 
     82 <ul>
     83 <li>Use {@link android.app.AlarmManager#setInexactRepeating setInexactRepeating()} instead
     84 of {@link android.app.AlarmManager#setRepeating setRepeating()} whenever possible.
     85 When you use {@link android.app.AlarmManager#setInexactRepeating setInexactRepeating()},
     86 Android synchronizes multiple inexact repeating alarms and fires
     87 them at the same time. This reduces the drain on the battery.</li>
     88 <li>If your alarm's behavior is based on an interval (for example, your alarm
     89 fires once an hour) rather than a precise trigger time (for example, your alarm fires at
     90 7 a.m. sharp and every 20 minutes after that), use an {@code ELAPSED_REALTIME}
     91 alarm type.</li>
     92 </ul></li>
     93 
     94 </ul>
     95 
     96 <h3 id="type">Choose an alarm type</h3>
     97 
     98 <p>One of the first considerations in using a repeating alarm is what its type should be.</p>
     99 
    100 
    101 <p>There are two general clock types for alarms: "elapsed real time" and "real time clock"
    102 (RTC).
    103 Elapsed real time uses the "time since system boot" as a
    104 reference, and real time clock uses UTC (wall clock) time. This means that
    105 elapsed real time is suited to setting an alarm based on the passage of time (for
    106 example, an alarm that fires every 30 seconds) since it isn't affected by
    107 time zone/locale. The real time clock type is better suited for alarms that are dependent
    108 on current locale.</p>
    109 
    110 <p>Both types have a "wakeup" version, which says to wake up the device's CPU if the
    111 screen is off. This ensures that the alarm will fire at the scheduled time. This is useful
    112 if your app has a time dependency&mdash;for example, if it has a limited window to perform a
    113 particular operation. If you don't use the wakeup version of your alarm type, then
    114 all the repeating alarms will fire when your device is next awake.</p>
    115 
    116 <p>If you simply need your alarm to fire at a particular interval (for example, every half
    117 hour), use one of the elapsed real time types. In general, this is the better choice.</p>
    118 
    119 <p>If you need your alarm to fire at a particular time of day,
    120 then choose one of the clock-based real time clock types. Note, however, that this approach can
    121 have some drawbacks&mdash;the app may not translate well to other locales, and if the user
    122 changes the device's time setting, it could cause unexpected behavior in your app.</p>
    123 
    124 <p>Here is the list of types:</p>
    125 
    126 <ul>
    127 
    128 <li>{@link android.app.AlarmManager#ELAPSED_REALTIME}&mdash;Fires the pending intent based
    129 on the amount of time since the  device was booted, but doesn't wake up the device. The
    130 elapsed time includes any time during which the device was asleep.</li>
    131 
    132 <li>{@link android.app.AlarmManager#ELAPSED_REALTIME_WAKEUP}&mdash;Wakes up the device and
    133 fires the pending intent after the specified length of time has elapsed since device
    134 boot.</li>
    135 
    136 <li>{@link android.app.AlarmManager#RTC}&mdash;Fires the pending intent
    137 at the specified time but does not wake up the device.</li>
    138 
    139 <li>{@link android.app.AlarmManager#RTC_WAKEUP}&mdash;Wakes up the
    140 device to fire the pending intent at the specified time.</li>
    141 </ul>
    142 
    143 <h4>ELAPSED_REALTIME_WAKEUP examples</h3>
    144 
    145 <p>Here are some examples of using {@link android.app.AlarmManager#ELAPSED_REALTIME_WAKEUP}.
    146 </p>
    147 
    148 <p>Wake up the device to fire the alarm in 30 minutes, and every 30 minutes
    149 after that:</p>
    150 
    151 <pre>
    152 // Hopefully your alarm will have a lower frequency than this!
    153 alarmMgr.setInexactRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP,
    154         AlarmManager.INTERVAL_HALF_HOUR,
    155         AlarmManager.INTERVAL_HALF_HOUR, alarmIntent);</pre>
    156 
    157 <p>Wake up the device to fire a one-time (non-repeating) alarm in one minute:</p>
    158 
    159 <pre>private AlarmManager alarmMgr;
    160 private PendingIntent alarmIntent;
    161 ...
    162 alarmMgr = (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
    163 Intent intent = new Intent(context, AlarmReceiver.class);
    164 alarmIntent = PendingIntent.getBroadcast(context, 0, intent, 0);
    165 
    166 alarmMgr.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,
    167         SystemClock.elapsedRealtime() +
    168         60 * 1000, alarmIntent);</pre>
    169 
    170 
    171 <h4>RTC examples</h3>
    172 
    173 <p>Here are some examples of using {@link android.app.AlarmManager#RTC_WAKEUP}.</p>
    174 
    175 <p>Wake up the device to fire the alarm at approximately 2:00 p.m., and repeat once a day
    176 at the same time:</p>
    177 
    178 <pre>// Set the alarm to start at approximately 2:00 p.m.
    179 Calendar calendar = Calendar.getInstance();
    180 calendar.setTimeInMillis(System.currentTimeMillis());
    181 calendar.set(Calendar.HOUR_OF_DAY, 14);
    182 
    183 // With setInexactRepeating(), you have to use one of the AlarmManager interval
    184 // constants--in this case, AlarmManager.INTERVAL_DAY.
    185 alarmMgr.setInexactRepeating(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(),
    186         AlarmManager.INTERVAL_DAY, alarmIntent);</pre>
    187 
    188 <p>Wake up the device to fire the alarm at precisely 8:30 a.m., and every 20 minutes
    189 thereafter:</p>
    190 
    191 <pre>private AlarmManager alarmMgr;
    192 private PendingIntent alarmIntent;
    193 ...
    194 alarmMgr = (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
    195 Intent intent = new Intent(context, AlarmReceiver.class);
    196 alarmIntent = PendingIntent.getBroadcast(context, 0, intent, 0);
    197 
    198 // Set the alarm to start at 8:30 a.m.
    199 Calendar calendar = Calendar.getInstance();
    200 calendar.setTimeInMillis(System.currentTimeMillis());
    201 calendar.set(Calendar.HOUR_OF_DAY, 8);
    202 calendar.set(Calendar.MINUTE, 30);
    203 
    204 // setRepeating() lets you specify a precise custom interval--in this case,
    205 // 20 minutes.
    206 alarmMgr.setRepeating(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(),
    207         1000 * 60 * 20, alarmIntent);</pre>
    208 
    209 <h3>Decide how precise your alarm needs to be</h3>
    210 
    211 <p>As described above, choosing the alarm type is often the first step in creating an alarm.
    212 A further distinction is how precise you need your alarm to be. For most apps,
    213 {@link android.app.AlarmManager#setInexactRepeating setInexactRepeating()} is the right
    214 choice.
    215 When you use this method, Android synchronizes multiple inexact repeating alarms and fires
    216 them at the same time. This reduces the drain on the battery.</p>
    217 
    218 <p>For the rare app that has rigid time requirements&mdash;for example, the alarm needs to
    219 fire precisely at 8:30 a.m., and every hour on the hour
    220 thereafter&mdash;use {@link android.app.AlarmManager#setRepeating setRepeating()}. But you
    221 should avoid using exact alarms if possible.</p>
    222 
    223 <p>With {@link android.app.AlarmManager#setInexactRepeating setInexactRepeating()},
    224 you can't specify a custom interval the way you can with
    225 {@link android.app.AlarmManager#setRepeating setRepeating()}. You have to use one of the
    226 interval constants, such as {@link android.app.AlarmManager#INTERVAL_FIFTEEN_MINUTES},
    227 {@link android.app.AlarmManager#INTERVAL_DAY}, and so on. See {@link android.app.AlarmManager}
    228 for the complete list.
    229 </p>
    230 
    231 <h2 id="cancel">Cancel an Alarm</h2>
    232 
    233 <p>Depending on your app, you may want to include the ability to cancel the alarm.
    234 To cancel an alarm, call {@link android.app.AlarmManager#cancel cancel()} on the Alarm
    235 Manager, passing in the {@link android.app.PendingIntent} you no longer want to fire. For
    236 example:</p>
    237 
    238 <pre>// If the alarm has been set, cancel it.
    239 if (alarmMgr!= null) {
    240     alarmMgr.cancel(alarmIntent);
    241 }</pre>
    242 
    243 <h2 id="boot">Start an Alarm When the Device Boots</h2>
    244 
    245 <p>By default, all alarms are canceled when a device shuts down.
    246 To prevent this from happening, you can design your application
    247 to automatically restart a repeating alarm if the user reboots the device. This ensures
    248 that the {@link android.app.AlarmManager} will continue doing its task without the user
    249 needing to manually restart the alarm.</p>
    250 
    251 
    252 <p>Here are the steps:</p>
    253 <ol>
    254 <li>Set the <a href="{@docRoot}reference/android/Manifest.permission.html#RECEIVE_BOOT_COMPLETED">
    255 {@code RECEIVE_BOOT_COMPLETED}</a> permission in your application's manifest. This allows
    256 your app to receive the
    257 {@link android.content.Intent#ACTION_BOOT_COMPLETED} that is broadcast after the system
    258 finishes booting (this only works if the app has already been launched by the user at least once):
    259 <pre>
    260 &lt;uses-permission android:name=&quot;android.permission.RECEIVE_BOOT_COMPLETED&quot;/&gt;</pre>
    261 </li>
    262 
    263 <li>Implement a {@link android.content.BroadcastReceiver} to receive the broadcast:
    264 <pre>public class SampleBootReceiver extends BroadcastReceiver {
    265 
    266     &#64;Override
    267     public void onReceive(Context context, Intent intent) {
    268         if (intent.getAction().equals(&quot;android.intent.action.BOOT_COMPLETED&quot;)) {
    269             // Set the alarm here.
    270         }
    271     }
    272 }</pre></li>
    273 
    274 <li>Add the receiver to your app's manifest file with an intent filter that filters on
    275 the {@link android.content.Intent#ACTION_BOOT_COMPLETED} action:
    276 
    277 <pre>&lt;receiver android:name=&quot;.SampleBootReceiver&quot;
    278         android:enabled=&quot;false&quot;&gt;
    279     &lt;intent-filter&gt;
    280         &lt;action android:name=&quot;android.intent.action.BOOT_COMPLETED&quot;&gt;&lt;/action&gt;
    281     &lt;/intent-filter&gt;
    282 &lt;/receiver&gt;</pre>
    283 
    284 
    285 <p>Notice that in the manifest, the boot receiver is set to
    286 {@code android:enabled=&quot;false&quot;}. This means that the receiver will not be called
    287 unless the application explicitly enables it. This prevents the boot receiver from being
    288 called unnecessarily. You can enable a receiver (for example, if the user sets an alarm)
    289 as follows:</p>
    290 
    291 <pre>ComponentName receiver = new ComponentName(context, SampleBootReceiver.class);
    292 PackageManager pm = context.getPackageManager();
    293 
    294 pm.setComponentEnabledSetting(receiver,
    295         PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
    296         PackageManager.DONT_KILL_APP);
    297 </pre>
    298 
    299 <p>Once you enable the receiver this way, it will stay enabled, even if the user reboots
    300 the device. In other words, programmatically enabling the receiver overrides the
    301 manifest setting, even across reboots. The receiver will stay enabled until your app disables it.
    302 You can disable a receiver (for example, if the user cancels an alarm) as follows:</p>
    303 
    304 <pre>ComponentName receiver = new ComponentName(context, SampleBootReceiver.class);
    305 PackageManager pm = context.getPackageManager();
    306 
    307 pm.setComponentEnabledSetting(receiver,
    308         PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
    309         PackageManager.DONT_KILL_APP);</pre>
    310 
    311 </li>
    312 </ol>
    313