Android5.0以上保活

Android5.0之后,很多都是可以被APP杀死,尤其是某机型做服务保活更难。Android5.0之后Android提供了JobService和JobScheduler这两的类。

当一系列预置的条件被满足时,JobScheduler API为你的应用执行一个操作。与AlarmManager不同的是这个执行时间是不确定的。除此之外,JobScheduler API允许同时执行多个任务。这允许你的应用执行某些指定的任务时不需要考虑时机控制引起的电池消耗。

1.创建Job Service

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

假定你使用的是Android Studio,当你点击了创建项目的完成按钮之后,你会得到一个”hello world”的应用骨架。你要做的第一步就是创建一个新的java类。为了简单起见,让我们创建一个继承自JobService且名字为JobSchedulerService的类,这个类必须实现两个方法,分别是onStartJob(JobParameters params)和 onStopJob(JobParameters params);

public class JobSchedulerService extends JobService {

@Override

public boolean onStartJob(JobParameters params) {

return false;

}

@Override

public boolean onStopJob(JobParameters params) {

return false;

}

}

当任务开始时会执行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定义的任务吧。

private Handler mJobHandler = new Handler( new Handler.Callback() {

@Override

public boolean handleMessage( Message msg ) {

Toast.makeText( getApplicationContext(),

"JobService task running", Toast.LENGTH_SHORT )

.show();

jobFinished( (JobParameters) msg.obj, false );

return true;

}

} );

在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)方法。

@Override

public boolean onStartJob(JobParameters params) {

mJobHandler.sendMessage( Message.obtain( mJobHandler, 1, params ) );

return true;

}

@Override

public boolean onStopJob(JobParameters params) {

mJobHandler.removeMessages( 1 );

return false;

}

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

android:permission="android.permission.BIND_JOB_SERVICE" />

2. 创建一个JobScheduler对象

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

mJobScheduler = (JobScheduler)

getSystemService( Context.JOB_SCHEDULER_SERVICE );

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

JobInfo.Builder builder = new JobInfo.Builder( 1,

new ComponentName( getPackageName(),

JobSchedulerService.class.getName() ) );

这个builder允许你设置很多不同的选项来控制任务的执行。下面的代码片段就是展示了如何设置以使得你的任务可以每隔三秒运行一次

builder.setPeriodic( 3000 );

其他设置方法 :

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中。

if( mJobScheduler.schedule( builder.build() ) <= 0 ) {

//If something goes wrong

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

如果你的应用想停止某个任务,你可以调用JobScheduler对象的cancel(int jobId)来实现;如果你想取消所有的任务,你可以调用JobScheduler对象的cancelAll()来实现。

mJobScheduler.cancelAll();

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

7.0 保活注意事项

// 构建JobInfo对象,传递给JobSchedulerService

JobInfo.Builder builder = new JobInfo.Builder(JOB_ID,new ComponentName(mContext, AliveJobService.class));

builder.setPeriodic(5000);

builder.setPersisted(true);

builder.setRequiresCharging(true);

JobInfo info = builder.build();

mJobScheduler.schedule(info);

这段定时任务在每隔5秒执行一次任务,Android 5.0和6.0系统能够正常运行.但是在Android7.0不能正常工作了。

解决办法

public void startJobScheduler() {

if (DEBUG) {

Log.i(TAG, "startJobScheduler");

}

int id = JOB_ID;

if (DEBUG) {

Log.i(TAG, "开启AstJobService id=" + id);

}

mJobScheduler.cancel(id);

JobInfo.Builder builder = new JobInfo.Builder(id, new ComponentName(mContext, AstJobService.class));

if (Build.VERSION.SDK_INT >= 24) {

builder.setMinimumLatency(JobInfo.DEFAULT_INITIAL_BACKOFF_MILLIS); //执行的最小延迟时间

builder.setOverrideDeadline(JobInfo.DEFAULT_INITIAL_BACKOFF_MILLIS); //执行的最长延时时间

builder.setMinimumLatency(JobInfo.DEFAULT_INITIAL_BACKOFF_MILLIS);

builder.setBackoffCriteria(JobInfo.DEFAULT_INITIAL_BACKOFF_MILLIS, JobInfo.BACKOFF_POLICY_LINEAR);//线性重试方案

} else {

builder.setPeriodic(JobInfo.DEFAULT_INITIAL_BACKOFF_MILLIS);

}

builder.setPersisted(true); // 设置设备重启时,执行该任务

builder.setRequiredNetworkType(JobInfo.NETWORK_TYPE_ANY);

builder.setRequiresCharging(true); // 当插入充电器,执行该任务

JobInfo info = builder.build();

mJobScheduler.schedule(info); //开始定时执行该系统任务

}

android job service,服务保活那些事(Job Service ,JobScheduler)相关推荐

  1. Android Service服务保活

    基于Android 10.11 实现的Service服务保活 最近公司要实现app程序不被杀死 实现持续定位,在网上看了很多个文章 也看到双进程保活,我忘了在那看到这篇文章了 下面 具体显现 创建 M ...

  2. Android 锁屏后Service服务保活(支持9.0)

    最近遇到个问题: 后台Service启动正常启动后,锁屏状态下大概80秒左右Service就被暂停了(并没有被杀死),唤醒屏幕后就继续执行. 解决方法: Service启动时创建一条通知,与其绑定,这 ...

  3. 【项目实战19】k8s(5)—service服务(IPVS均衡负载)

    service服务 一.前言 (1).service简述 (2).service类型 二.IPVS服务 (1).简介 (2).使用 三.service外部访问方式 (1).NodePort (2).L ...

  4. Android Service服务不被杀死(保活)

    1.设置->应用->运行中->停止->杀死service 这样可以在service的onDestroy()方法中重启service public void onDestroy( ...

  5. android8.1启动前台服务,Android - 保活(1)前台服务保活

    老婆保佑,代码无BUG 前言 项目中遇到一个需求,需要竟可能的上传用户的定位信息,引发了我对目前已知的保活手段的探究,同时也遇到过客户说,推送不能收到,不能像微信那样,MMP的,不想理客户 目录 一: ...

  6. Android Service服务的相关介绍

    文章目录 Android Service服务的相关介绍 创建方式 启动方式 生命周期 onStartCommand参数及返回值的理解 ServiceConnection 场景说明 前台服务 问答 st ...

  7. android四大组件 服务,Android四大组件之Service

    Service Service(服务)是一个可以在后台执行长时间运行操作而不使用用户界面的应用组件.服务可由其他应用组件启动,而且即使用户切换到其他应用,服务仍将在后台继续运行. 此外,组件可以绑定到 ...

  8. 【Android 进程保活】应用进程拉活 ( JobScheduler 拉活 | JobScheduler 使用流程 | JobService 服务 | 不同版本兼容 | 源码资源 )

    文章目录 一. JobScheduler 用法简介 二. JobScheduler 拉活完整代码 1. JobService 2.清单文件 3.启动 JobScheduler 任务 4.运行效果 三. ...

  9. Android Service 服务(二)—— BroadcastReceiver

    一. BroadcastReceiver简介 BroadcastReceiver,用于异步接收广播Intent,广播Intent是通过调用Context.sendBroadcast()发送.Broad ...

最新文章

  1. watchdog应用实例
  2. Dapper,大规模分布式系统的跟踪系统--转
  3. VMware / 三种联网方法及原理
  4. 牛客练习赛 60(待补E-长链剖分或者dsu)
  5. MTK 驱动(67)---深入MTK平台bootloader启动之【 lk -amp;gt; kernel】分析笔记
  6. DevOps使用教程 华为云(10)GitHub git Pull Request 合并请求
  7. 精英主义 遗传算法 matlab,遗传算法matlab程序
  8. spring揭秘总结(一)——spring的Ioc容器
  9. 使用NOKIA MMS LIBRARY发送中国移动彩信
  10. 数据聚合与分组操作(数据分析)
  11. 【码上实战】【立体匹配系列】经典AD-Census: (5)扫描线优化
  12. 8421BCD码的加6修正的原因
  13. matlab求解整数规划问题
  14. 报告|中国智能音箱已入局全球市场,双重商业模式迅速扩张
  15. 2.Enable ADB integration' to be enabled.
  16. C++控制台贪吃蛇小游戏详细教程
  17. 一个疯子的梦-20190921
  18. go语言ORM框架ent使用教程
  19. 无忧无盘服务器内存,无忧无盘安装以及配置教材.doc
  20. 论文笔记- AdaDepth: Unsupervised Content Congruent Adaptation for Depth Estimation

热门文章

  1. TokyoTyrant的管理工具tcrmgr使用小记
  2. 条款23: 必须返回一个对象时不要试图返回一个引用
  3. 把字符串复制到剪贴板
  4. 方差、标准差(均方差),均方误差、均方根误差
  5. 151. 反转字符串里的单词
  6. 大数据分析有哪些特征
  7. 大数据分析技术架构的通用模块
  8. Tensorflow2.0入门教程(一)
  9. MVC和MTV初步认识+django的一个简单应用(萌新交流互动,欢迎大家指出错误)
  10. jsp获取连接池的实时连接数_数据库连接池原理分析及模拟实现