在这篇文章中,你会学习到在Android 5.0中怎样使用JobScheduler API。

JobScheduler API同意开发人员在符合某些条件时创建运行在后台的任务。

介绍

在Android开发中,会存在这么些场景 : 你须要在稍后的某个时间点或者当满足某个特定的条件时运行一个任务,比如当设备接通电源适配器或者连接到WIFI。幸运的是在API 21 ( Android 5.0。即Lollipop )中,google提供了一个新叫做JobScheduler API的组件来处理这种场景。

当一系列预置的条件被满足时,JobScheduler API为你的应用运行一个操作。与AlarmManager不同的是这个运行时间是不确定的。除此之外。JobScheduler API同意同一时候运行多个任务。

这同意你的应用运行某些指定的任务时不须要考虑时机控制引起的电池消耗。

这篇文章中,你会学到关于JobScheduler API很多其它的东西以及在你的应用中用于运行一个简单的后台任务的JobService,这篇文章中所展示的代码你都能够在github中找到。

1. 创建Job Service

首先,你须要创建一个API最低为21的Android项目,因此JobScheduler是近期的版本号才加入Android的,在写这篇文章的时候,它还没有兼容库支持。

假定你使用的是Android Studio,当你点击了创建项目的完成button之后,你会得到一个”hello world”的应用骨架。你要做的第一步就是创建一个新的java类。

为了简单起见。让我们创建一个继承自JobService且名字为JobSchedulerService的类,这个类必须实现两个方法。各自是onStartJob(JobParameters params)onStopJob(JobParameters params)

  1. public class JobSchedulerService extends JobService {
  2.  
  3. @Override
  4. public boolean onStartJob(JobParameters params) {
  5.  
  6. return false;
  7. }
  8.  
  9. @Override
  10. public boolean onStopJob(JobParameters params) {
  11.  
  12. return false;
  13. }
  14. }

当任务開始时会运行onStartJob(JobParameters params)方法,由于这是系统用来触发已经被运行的任务。

正如你所示,这种方法返回一个boolean值。假设返回值是false,系统假设这种方法返回时任务已经运行完成。假设返回值是true,那么系统假定这个任务正要被运行,运行任务的重担就落在了你的肩上。

当任务运行完成时你须要调用jobFinished(JobParameters params, boolean needsRescheduled)来通知系统。

当系统接收到一个取消请求时,系统会调用onStopJob(JobParameters params)方法取消正在等待运行的任务。非常重要的一点是假设onStartJob(JobParameters params)返回false,那么系统假定在接收到一个取消请求时已经没有正在运行的任务。换句话说,onStopJob(JobParameters params)在这种情况下不会被调用。

须要注意的是这个job service运行在你的主线程,这意味着你须要使用子线程,handler, 或者一个异步任务来运行耗时的操作以防止堵塞主线程。

由于多线程技术已经超出了我们这篇文章的范围。让我们简单实现一个Handlder来运行我们在JobSchedulerService定义的任务吧。

  1. private Handler mJobHandler = new Handler( new Handler.Callback() {
  2.  
  3. @Override
  4. public boolean handleMessage( Message msg ) {
  5. Toast.makeText( getApplicationContext(),
  6. "JobService task running", Toast.LENGTH_SHORT )
  7. .show();
  8. jobFinished( (JobParameters) msg.obj, false );
  9. return true;
  10. }
  11.  
  12. } );

在Handler中,你须要实现handleMessage(Message msg)方法来处理你的任务逻辑。在这个样例中。我们尽量保证样例简单,因此我们仅仅在handleMessage(Message msg)中显示了一个Toast。这里就是你要写你的任务逻辑( 耗时操作 )的地方,比方同步数据等。

当任务运行完成之后,你须要调用jobFinished(JobParameters params, boolean needsRescheduled)来让系统知道这个任务已经结束,系统能够将下一个任务加入到队列中。假设你没有调用jobFinished(JobParameters params, boolean needsRescheduled),你的任务仅仅会运行一次,而应用中的其它任务就不会被运行。

jobFinished(JobParameters params, boolean needsRescheduled)的两个參数中的params參数是从JobService的onStartJob(JobParameters params)的params传递过来的。needsRescheduled參数是让系统知道这个任务是否应该在最处的条件下被反复运行。这个boolean值非常实用,由于它指明了你怎样处理由于其它原因导致任务运行失败的情况,比如一个失败的网络请求调用。

创建了Handler实例之后。你就能够实现onStartJob(JobParameters params) 和onStopJob(JobParameters params)方法来控制你的任务了。

你可能已经注意到在以下的代码片段中onStartJob(JobParameters params)返回了true。这是由于你要通过Handler实例来控制你的操作,

这意味着Handler的handleMessage方法的运行时间可能比onStartJob(JobParameters params)更长。返回true,你会让系统知道你会手动地调用jobFinished(JobParameters params, boolean needsRescheduled)方法。

  1. @Override
  2. public boolean onStartJob(JobParameters params) {
  3. mJobHandler.sendMessage( Message.obtain( mJobHandler, , params ) );
  4. return true;
  5. }
  6.  
  7. @Override
  8. public boolean onStopJob(JobParameters params) {
  9. mJobHandler.removeMessages( );
  10. return false;
  11. }

一旦你在Java部分做了上述工作之后,你须要到AndroidManifest.xml中加入一个service节点让你的应用拥有绑定和使用这个JobService的权限。

  1. <service android:name=".JobSchedulerService"
  2. android:permission="android.permission.BIND_JOB_SERVICE" />

2. 创建一个JobScheduler对象

随着JobSchedulerService构建完成,我们能够開始研究你的应用怎样与JobScheduler API进行交互了。第一件要做的事就是你须要创建一个JobScheduler对象,在实例代码的MainActivity中我们通过getSystemService( Context.JOB_SCHEDULER_SERVICE )初始化了一个叫做mJobScheduler的JobScheduler对象。

  1. mJobScheduler = (JobScheduler)
  2. getSystemService( Context.JOB_SCHEDULER_SERVICE );

当你想创建定时任务时。你能够使用JobInfo.Builder来构建一个JobInfo对象,然后传递给你的Service。JobInfo.Builder接收两个參数。第一个參数是你要运行的任务的标识符。第二个是这个Service组件的类名。

  1. JobInfo.Builder builder = new JobInfo.Builder( ,
  2. new ComponentName( getPackageName(),
  3. JobSchedulerService.class.getName() ) );

这个builder同意你设置非常多不同的选项来控制任务的运行。

以下的代码片段就是展示了怎样设置以使得你的任务能够每隔三秒运行一次。

  1. builder.setPeriodic( );

其它设置方法 :

  • setMinimumLatency(long minLatencyMillis): 这个函数能让你设置任务的延迟运行时间(单位是毫秒),这个函数与setPeriodic(long time)方法不兼容,假设这两个方法同一时候调用了就会引起异常;
  • setOverrideDeadline(long maxExecutionDelayMillis): 
    这种方法让你能够设置任务最晚的延迟时间。假设到了规定的时间时其它条件还未满足。你的任务也会被启动。与setMinimumLatency(long time)一样,这种方法也会与setPeriodic(long time),同一时候调用这两个方法会引发异常。
  • setPersisted(boolean isPersisted): 
    这种方法告诉系统当你的设备重新启动之后你的任务是否还要继续运行。
  • setRequiredNetworkType(int networkType): 
    这种方法让你这个任务仅仅有在满足指定的网络条件时才会被运行。默认条件是JobInfo.NETWORK_TYPE_NONE。这意味着无论是否有网络这个任务都会被运行。

    另外两个可选类型。一种是JobInfo.NETWORK_TYPE_ANY,它表明须要随意一种网络才使得任务能够运行。

    还有一种是JobInfo.NETWORK_TYPE_UNMETERED,它表示设备不是蜂窝网络( 比方在WIFI连接时 )时任务才会被运行。

  • setRequiresCharging(boolean requiresCharging): 
    这种方法告诉你的应用,仅仅有当设备在充电时这个任务才会被运行。
  • setRequiresDeviceIdle(boolean requiresDeviceIdle): 
    这种方法告诉你的任务仅仅有当用户没有在使用该设备且有一段时间没有使用时才会启动该任务。

须要注意的是setRequiredNetworkType(int networkType)setRequiresCharging(boolean requireCharging) and setRequiresDeviceIdle(boolean requireIdle)者几个方法可能会使得你的任务无法运行,除非调用setOverrideDeadline(long time)设置了最大延迟时间,使得你的任务在为满足条件的情况下也会被运行。

一旦你预置的条件被设置,你就能够构建一个JobInfo对象。然后通过例如以下所示的代码将它发送到你的JobScheduler中。

  1. if( mJobScheduler.schedule( builder.build() ) <= ) {
  2. //If something goes wrong
  3. }

你可能注意到了。这个schedule方法会返回一个整型。假设schedule方法失败了。它会返回一个小于0的错误码。否则它会我们在JobInfo.Builder中定义的标识id。

假设你的应用想停止某个任务。你能够调用JobScheduler对象的cancel(int jobId)来实现;假设你想取消全部的任务。你能够调用JobScheduler对象的cancelAll()来实现。

  1. mJobScheduler.cancelAll();

到了这里,你如今应该已经知道怎样在你的应用中使用JobScheduler API来运行批量任务和后台操作了。

结论

这篇文章中,你学会了怎么实现一个使用Handler对象来运行后台任务的JobService子类,你也学会了怎样使用JobInfo.Builder来设置JobService。

掌握了这些之后,你能够在降低资源消耗的同一时候提升应用的效率。

Android 5.0中使用JobScheduler的更多相关文章

  1. 在Android 5.0中使用JobScheduler

    在Android 5.0中使用JobScheduler 原文链接 : using-the-jobscheduler-api-on-android-lollipop 译者 : Mr.Simple 校对者 ...

  2. 在Android 5.0中使用JobScheduler(转载)

    翻译见:http://blog.csdn.net/bboyfeiyu/article/details/44809395 In this tutorial, you will learn how to ...

  3. Android 7.0 中 ContentProvider 实现原理

    欢迎大家前往腾讯云社区,获取更多腾讯海量技术实践干货哦~ 作者:汪毅雄 导语: 本文描述了ContentProvider发布者和调用者这两在Framework层是如何实现的. 作为Android的四大 ...

  4. Android 6.0 中的 Wifi 连接

    Android 6.0 中的 Wifi 连接 这几天在写一个软件,结果被其中的 wifi 连接问题困扰了 3 天. 先描述下需求: usb 接口接了一根 usb2serial,通过这个接口接收命令 当 ...

  5. 我的Android进阶之旅------>如何解决Android 5.0中出现的警告: Service Intent must be explicit:

    我的Android进阶之旅-->如何解决Android 5.0中出现的警告: java.lang.IllegalArgumentException: Service Intent must be ...

  6. 我的Android进阶之旅------&gt;怎样解决Android 5.0中出现的警告: Service Intent must be explicit:

    我的Android进阶之旅-->怎样解决Android 5.0中出现的警告: java.lang.IllegalArgumentException: Service Intent must be ...

  7. android 4.0 中出错 java.lang.UnsupportedOperationException

    在android4.0中  画图的时候使用: canvas.clipPath(path, Region.Op.XOR); 报错 java.lang.UnsupportedOperationExcept ...

  8. Android Studio3.0中dependencies依赖由compile变为implementation的区别

    前言 Android Studio版本更新至3.0了,更新后,连带着com.android.tools.build:gradle 工具也升级到了3.0.0,在3.0.0中使用了最新的Gralde 4. ...

  9. Android 6.0 中TimePicker显示为滚动样式的方法

    在Android6.0中,TimePicker控件的默认样式为转盘的样式,就像这个样子: 如果想要显示为之前的滚动样式的话也很简单,只要在布局文件中设置TimePicker的timePickerMod ...

随机推荐

  1. Android 多线程断点续传同时下载多个大文件

    最近学习在Android环境中一些网络请求方面的知识,其中有一部分是关于网络下载方面的知识.在这里解析一下自己写的demo,总结一下自己所学的知识.下图为demo的效果图,仿照一些应用下载商城在Lis ...

  2. 记阮一峰---JavaScript 标准参考教程之标准库-Object对象

    在看到阮大神的-标准库-Object对象时 有个 类型判断类型 方法可能以后会用到.特此记录一下 4.3:toString()的应用:判断数据类型 Object.prototype.toString方 ...

  3. pytest_多用例执行(1)

    一.首先创建测试套件 # -*- coding:utf-8 -*-from __future__ import print_functionimport pytestimport allure cla ...

  4. System and method for critical address space protection in a hypervisor environment

    A system and method in one embodiment includes modules for detecting an access attempt to a critical ...

  5. Solr的关键特性

    1.基于标准的开放接口:Solr搜索服务器支持通过XML.JSON和HTTP查询和获取结果. 2.易管理:Solr可以通过HTML页面管理,Solr配置通过XML完成. 3.可伸缩性:能够有效地复制到 ...

  6. USB串行端口

    USB-SERIAL CH341A(COM22)USB串行端口

  7. iOS 之应用性能调优的25个建议和技巧

    注意:每在优化代码之前,你都要注意一个问题,不要养成"预优化"代码的错误习惯. 时常使用Instruments去profile你的代码来发现须要提升的方面.Matt Gallowa ...

  8. [Angular] Using the Argon 2 Hashing Function In Our Sign Up Backend Service

    Which hash algorithom to choose for new application: https://www.owasp.org/index.php/Password_Storag ...

  9. [Angular] Use Angular’s @HostBinding and :host(...) to add styling to the component itself

    One thing that we can do is to add styles directly to HTML elements that live within our component. ...

  10. angular 响应式自定义表单控件—注册头像实例

    1. 组件继承ControlValueAccessor,ControlValueAccessor接口需要实现三个必选方法 writeValue() 用于向元素中写入值,获取表单的元素的元素值 regi ...