Android 的定时任务
1,Java的API提供的Timer类
Android 中使用容易受手机的休眠系统影响(例如:手机休眠,导致了这个功能的停止)。


2,Android的Alarm机制
Alarm 机制:  主要就是借助了AlarmManager 类来实现的。这个类和NotificationManager 有点类似,都是通过调用Context 的
getSystemService()方法来获取实例的,只是这里需要传入的参数是Context.ALARM_SERVICE。因此,获取一个AlarmManager 的实例就可以写成:

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

定时任务设置:(例如设定任务10s执行)

long triggerAtTime = SystemClock.elapsedRealtime() + 10 * 1000;manager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,triggerAtTime,pendingIntent);

参数意义:
使用SystemClock.elapsedRealtime()方法可以获取到系统开机至今所经历时间的毫秒数,使用System.currentTimeMillis()方法可以获取到1970 年1 月1 日0 点至今所经历时间的毫秒数。

第一个参数有四个值可以选
------ELAPSED_REALTIME
ELAPSED_REALTIME 表示让定时任务的触发时间从系统开机开始算起,但不会唤醒CPU

------ELAPSED_REALTIME_WAKEUP
ELAPSED_REALTIME_WAKEUP 同样表示让定时任务的触发时间从系统开机开始算起,但会唤醒CPU

------RTC
RTC 表示让定时任务的触发时间从1970 年1月1 日0 点开始算起,但不会唤醒CPU

------RTC_WAKEUP
RTC_WAKEUP 同样表示让定时任务的触发时间从1970 年1 月1 日0 点开始算起,但会唤醒CPU

第二个参数
就是定时任务触发的时间,以毫秒为单位。
如果第一个参数使用的是ELAPSED_REALTIME 或ELAPSED_REALTIME_WAKEUP,则这里传入开机至今的时间再加上延迟执行的时间。如果第一个参数使用的是RTC 或RTC_WAKEUP,则这里传入1970 年1 月1 日0 点至今的时间再加上延迟执行的时间。


第三个参数
PendingIntent
般会调用getBroadcast()方法来获取一个能够执行广播的PendingIntent。这样当定时任务被触发的时候,广播接收器的onReceive()方法就可以得到执行。


设定一个任务在10 秒钟后执行还可以写成:

long triggerAtTime = System.currentTimeMillis() + 10 * 1000;manager.set(AlarmManager.RTC_WAKEUP, triggerAtTime, pendingIntent);

创建一个可以长期在后台执行定时任务的服务

public class LongRunningService extends Service {@Overridepublic IBinder onBind(Intent intent) {return null;}@Overridepublic int onStartCommand(Intent intent, int flags, int startId) {new Thread(new Runnable() {@Overridepublic void run() {Log.d("LongRunningService", "executed at " + new Date().toString());}}).start();AlarmManager manager = (AlarmManager) getSystemService(ALARM_SERVICE);int anHour = 60 * 60 * 1000; // 这是一小时的毫秒数long triggerAtTime = SystemClock.elapsedRealtime() + anHour;Intent i = new Intent(this, AlarmReceiver.class);PendingIntent pi = PendingIntent.getBroadcast(this, 0, i, 0);manager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP, triggerAtTime, pi);return super.onStartCommand(intent, flags, startId);}}

onStartCommand()方法里开启了一个子线程,然后在子线程里就可以执行具体的逻辑操作了。

新建一个AlarmReceiver 类,并让它继承自BroadcastReceiver,代码如下所示:

public class AlarmReceiver extends BroadcastReceiver {@Overridepublic void onReceive(Context context, Intent intent) {Intent i = new Intent(context, LongRunningService.class);context.startService(i);}}

循环流程
启动Service ----->onStartCommand()(执行定时任务(并发送广播)----->AlarmReceiver(接收广播,再次启动Service)-----
          |                                                                                                                                                                                                                                     -
         --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

注意:
从Android 4.4 版本开始,Alarm 任务的触发时间将会变得不准确,有可能会延迟一段时间后任务才能得到执行。这并不是个bug,而是系统在耗电性方面进行的优化。系统会自动检测目前有多少Alarm 任务存在,然后将触发时间将近的几个任务放在一起执行,这就可以大幅度地减少CPU 被唤醒的次数,从而有效延长电池的使用时间。当然,如果你要求Alarm 任务的执行时间必须准备无误,Android 仍然提供了解决方案。使用AlarmManager 的setExact()方法来替代set()方法,就可以保证任务准时执行了。

更多相关文章

  1. android和ios的系统特性区别
  2. Android编译系统-上
  3. Android系统Surface机制的SurfaceFlinger服务渲染应用程序UI的过
  4. Android系统利用tcpdump抓包
  5. android 系统服务
  6. Android 应用程序(APK) 如何获得系统签名权限 强制关闭程序(后台
  7. Android 9 (P)系统启动之SystemServer大揭秘下

随机推荐

  1. Java jni 开发
  2. android最新源码(4.4.2_r1版本以上)下载
  3. 设置系统超时时间
  4. android studio添加忽略文件
  5. 第100章、WebView应用之Javascript调用An
  6. 根据Android架构分层推荐开发书籍
  7. Android---1---HelloWorld
  8. Android(安卓)Camera 使用小结
  9. TextView中ellipsize属性
  10. 文章分享:Android四大组件详解