Android 的 JobScheduler
介绍
在之前有介绍过Android O 中的后台执行限制中,增加了"前台"与"后台"的概念。对于处于"后台"的服务系统可能会将其关闭。对此,官方推荐使用 JobScheduler 来代替某些"后台"的服务。
对于 Android 的 JobScheduler ,最早是在 Android 5.0 中提供的 api,,顾名思义,调度作业,使用这个 api 就是把作业(JobInfo)的调度权交给系统执行。该框架收到回调,会试图尽可能地批次和推迟作业的执行(以此来减少电池消耗,达到省电的目的)。通常,如果不给作业指定一个期限,它可以 jobscheduler的内部队列中的任何时刻运行,也可能被推迟,甚至可以推迟到下一次设备重新启动。我们也可以设定一些条件,如网络变化,充电插拔,周期执行,达到指定条件,系统会执行作业。
使用
JobScheduler 是系统的服务,所以需要是使用方法getSystemService(Context.JOB_SCHEDULER_SERVICE) 获取实例:
JobScheduler js =(JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);
首先,我们需要一个建立一个 JobService 作为执行的任务。
public class MyJobService extends JobService {
public MyJobService() {
}
@Override
public boolean onStartJob(JobParameters jobParameters) {
return false;
}
@Override
public boolean onStopJob(JobParameters jobParameters) {
return false;
}
}
在文件 AndroidManifest.xml 中注册该服务,其中服务权限android:permission="android.permission.BIND_JOB_SERVICE"是必须的:
<service
android:name=".MyJobService"
android:permission="android.permission.BIND_JOB_SERVICE"
android:exported="true" />
当任务开始时会执行onStartJob(JobParameters params)方法,因为这是系统用来触发已经被执行的任务。这个方法返回一个boolean值,如果返回值是false,系统假设这个方法返回时任务已经执行完毕。如果返回值是true,那么系统假定这个任务正要被执行,当任务执行完毕时你需要调用jobFinished(JobParameters params, boolean needsRescheduled)来通知系统。
当系统接收到一个取消请求时,系统会调用onStopJob(JobParameters params)方法取消正在等待执行的任务。如果onStartJob(JobParameters params)返回false,那么系统假定在接收到一个取消请求时已经没有正在运行的任务。换句话说,onStopJob(JobParameters params)在这种情况下不会被调用。
需要注意的是这个job service运行在你的主线程,这意味着你需要使用子线程,handler, 或者一个异步任务来运行耗时的操作以防止阻塞主线程。
创建好JobService,需要创建一个JobInfo 类来设置所调度任务的调度条件,JobInfo 提供了建造者模式 —JobInfo.Builder,并在其中设置任务的调度条件 :
JobInfo.Builder builder = new JobInfo.Builder(1, new ComponentName(this, MyJobService.class));
// 设置当设备接入不计量的网络,通常就是指wifi
builder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_UNMETERED);
//最后使用JobInfo.Builder的build() 方法生成最后提交给 jobscheduler 的JobInfo
JobInfo jobInfo = builder.build();
对于任务的调度条件还有如下(以下只是部分,全部的请参考官网的 API JobInfo.Builder ):
//设置周期执行的时间
setPeriodic(long time)
//设置任务的延迟执行时间(单位是毫秒),这个函数与方法不兼容,如果这两个方法同时调用了就会引起异常;
setMinimumLatency(long minLatencyMillis):
//设置任务最晚的延迟时间。如果到了规定的时间时其他条件还未满足,你的任务也会被启动。与setMinimumLatency(long time)一样,这个方法也会与setPeriodic(long time),同时调用这两个方法会引发异常。
setOverrideDeadline(long maxExecutionDelayMillis):
//告诉系统当设备重启之后你的任务是否还要继续执行。
setPersisted(boolean isPersisted):
//让这个任务只有在满足指定的网络条件时才会被执行。默认条件是JobInfo.NETWORK_TYPE_NONE,这意味着不管是否有网络这个任务都会被执行。另外两个可选类型,一种是JobInfo.NETWORK_TYPE_ANY,它表明需要任意一种网络才使得任务可以执行。另一种是JobInfo.NETWORK_TYPE_UNMETERED,它表示设备不是蜂窝网络( 比如在WIFI连接时 )时任务才会被执行。
setRequiredNetworkType(int networkType):
//告诉应用只有当设备在充电时这个任务才会被执行。
setRequiresCharging(boolean requiresCharging):
//任务只有当用户没有在使用该设备且有一段时间没有使用时才会启动该任务。
setRequiresDeviceIdle(boolean requiresDeviceIdle):
构建好 JobService和 jobInfo,就可以把任务交给 JobScheduler 就可以了,等到条件达成,系统就可以自动执行任务了:
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
JobScheduler js =(JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);
JobInfo.Builder builder = new JobInfo.Builder(1, new ComponentName(this, MyJobService.class));
builder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_UNMETERED);
JobInfo jobInfo = builder.build();
js.schedule(jobInfo);
}
}
总结
1.通过以上就可以实现自己的 JobService 并交由系统在达到指定条件的时候自动调度了,通过 JobScheduler 可以改善自己的应用程序在功耗上问题。