Android的定时任务有两种,一种是JavaAPI提供的Timer类,另一种就是使用Android的Alarm机制。

Timer不太适合那些需要长期在后台运行的定时任务,因为每个手机都有自己的休眠策略,Android手机长时间不操作就会导致Timer定时任务无法执行,而Alarm具有唤醒CPU的功能,能保证大多数情况下,执行定时任务的时候CPU能正常工作。

AlarmManager manager= (AlarmManager) getSystemService(Context.ACTIVITY_SERVICE);
 PendingIntent pendingIntent;


      /**
         *    SystemClock.elapsedRealtime():获取到系统开机至今所经历的毫秒数
         *    System.currentTimeMillis():1970-1-1 0:00至今所经历的毫秒数
         *
         *     ELAPSED_REALTIME:定时任务触发从系统开机算起 但不唤醒CPU
         *     ELAPSED_REALTIME_WAKEUP:定时任务触发从系统开机算起 唤醒CPU
         *     RTC:从1970-1-1 0:00算起 不唤醒CPU
         *     RTC_WAKEUP:从1970-1-1 0:00算起 唤醒CPU
         */

    /**pendIntent:一般会调用getSrervice或getBroadCast
       来获取一个能够执行服务或广播的pendingIntent,
       这样才能保证在任务被触发的时候,
       服务里的onStartCommand()和onRecive()方法被执行   
      */


   long time= SystemClock.elapsedRealtime()+10*1000;//10秒钟后执行任务
   manager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,time,pendingIntent);

那么要实现一个长时间在后台定时运行的服务该怎么做?

创建一个MyService类

package com.example.test;

import android.app.AlarmManager;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.SystemClock;

import androidx.annotation.Nullable;

public class MyService extends Service {


    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {

        new Thread(new Runnable() {
            @Override
            public void run() {
                //TODO
            }
        }).start();

        AlarmManager manager= (AlarmManager) getSystemService(ALARM_SERVICE);
        int time=60*60*1000;//一小时

        long triggerAtTime= SystemClock.elapsedRealtime()+time;
        Intent intent1=new Intent(this,MyService.class);
        PendingIntent pendingIntent=PendingIntent.getService(this,0,intent1,0);
        manager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,triggerAtTime,pendingIntent);

        return super.onStartCommand(intent, flags, startId);
    }
}

可以看到先创建了一个子线程在这里执行具体的操作任务,然后就是alarm机制,定义任务触发一小时后,再使用PendingIntent指定定时处理任务的MyService,最后调用set方法设定完成。

这样就将一个长时间在后台定时运行的服务设定成功了,一旦启动了MyService,就会在onStartCommand()方法里设定一个定时任务,这样一小时后将会再次启动MyService,每隔一小时就会执行一次。

然后去使用,在manifest文件中注册服务

<service android:name=".MyService"/>

在使用的地方调用下面代码就可以。

Intent intent=new Intent(context,MyService.class);
    context.startService(intent);

但是在Android4.4后,定时任务变得不准确,会延时一段时间才能执行,那是因为系统在耗电方面进行的优化,系统会检测有多少Alarm任务存在,将触发时间相近的几个任务放在一起执行,这样可以大幅度的减少CPU被唤醒的次数,演唱电池的使用时长。也可以使用setExact()来代替set()方法,可以保证任务能够准时执行。

Doze模式:

如果设备未充电,并处于静止状态,且屏幕关闭了一段时间后,就能进入到Doze模式。

在Doze模式下,系统会对CPU、网络、Alarm等活动进行限制,从而延长电池的使用寿命。当然,系统并不会一直处于Doze模式,而是会间歇性地退出Doze模式一小段时间,在这段时间中,应用就可以去完成它的同步操作、Alarm任务等。

Android AlarmManager 设置时间不起作用 android alarmanager 定时不管用_android

 

Doze模式下受限制的功能:

  • 网络访问被禁止。
  •     系统忽略唤醒CPU或者屏幕操作。
  •     系统不再执行WIFI扫描。
  •     系统不再执行同步服务。
  •     Alarm任务将会在下次退出Doze模式的时候执行。

在Doze模式下, Alarm任务会变得相当不准,当然这种情况下对Alarm的要求也并不高,如果有特殊的要求,调用AlarmManager的setAndAllowWhileIdle()或者setExactAndAllowWhileIdle()方法就能让定时任务正常执行,和set()与setExact()的区别是一样的。