Android Fk: JobService的使用解析

2024-03-19 09:58

本文主要是介绍Android Fk: JobService的使用解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

某个时间点或者满足某个特定的条件时去执行某个任务时可以使用JobService来实现,一从API19起,使用AlarmManager,setRepeating定时请求并不能保证时效的准确,在5.0以后,Google推出了一个JobService,用来执行一些并非即时执行的后台进程。

一、使用方法及注意事项:

1.新建自己的JobService

public class DemoJobService extends JobService
{@Overridepublic boolean onStartJob(JobParameters params) {return false;}@Overridepublic boolean onStopJob(JobParameters params) {return false;}
} 

2. schedule这个service

JobScheduler scheduler =(JobScheduler) context.getSystemService(Context.JOB_SCHEDULER_SERVICE);
ComponentName componentName =new ComponentName(context, DemoJobService.class);  
JobInfo.Builder builder =new JobInfo.Builder(DemoJobService.INITJOBID, componentName);  
builder.setMinimumLatency(1000);
scheduler.cancel(ScreenPaperModeUpdateService.INITJOBID);
scheduler.schedule(builder.build()); 

3. 重要回调的说明

API调用时机作用备注
onStartJob()Job开始时候的回调实现具体的工作逻辑return false 系统将会在逻辑结束后自动结束本job
onStopJob()当JobScheduler感知到Job条件不满足的时候,或者Job被抢占被取消的时候强制回调停止该Job如果希望在这种意外情况下重启Job,返回值应该返回true
jobFinished()当任务完成的时候,客户端主动调用,通知JobScheduler已经完成了任务完成任务销毁Job调用该方法的时候不会回调onStopJob(),但会回调onDestroy
onCreate()Service被初始化的时候回调初始化可以用来设置BroadcastReceiver或者ContentObserver
onDestroy()Service被销毁时候的回调用来注销BroadcastReceiver或者ContentObserver,释放资源等

4. 注意事项:

结合同事整理的笔记:

  1. AndroidManifest.xml中要声明Service ,Manifest文件里声明JobService的时候必须要请求android:permission="android.permission.BIND_JOB_SERVICE"的权限。否则在schedule的时候会抛出IllegalArgumentException异常 Error: requested job be persisted without holding RECEIVE_BOOT_COMPLETED permission
  2. JobInfo创建的时候必须至少设置一个条件,否则在创建的时候就会抛出IllegalArgumentException异常,You’re trying to build a job with no constraints, this is not allowed
  3. jobFinished(),如果该jobservice是个耗时操作,onStartJob中需要return true()在return之前需要主动调用jobFinished()方法告诉JobSchedulerService 该Jobservice执行结束可以走销毁流程了。
  4. 同一个UID进程里只能有唯一一个Job的ID,否则新生成的Job会抢占已经运行的Job,导致该Job被异常终止
  5. JobService因为设置的条件变化导致被强制停止后,如果需要自重启Service,需要将onStopJob返回true
  6. 自行cancel了Job,即便onStopJob返回true也不会再启动,这种时候如果需要自启动,需要在onStopJob里重新schedule
  7. 自行finished了Job,onStopJob不会回调,只会回调onDestroy
  8. Job如果需要执行长时间任务的话,onStartJob应该返回true,否则onStartJob刚回调结束,Job就会被停止
  9. 使用CancelAll会cancel同一UID下的所有job
  10. ENFORCE_MAX_JOBS =true,同个UID最多100个Job
  11. JobService是不能保证精确准时的

针对第8条,因为涉及到在onStartJob里运行耗时操作和异步线程的问题,有许多注意事项:
首先特别强调,如果onStartJob()中返回了false,内部开启了工作线程去处理逻辑,这个时候Job会被强制销毁,同时后台任务还在继续,如果我们在onDestroy中没有释放线程,这个时候就会出现内存泄露,所以这种写法一定是要杜绝的

  • onStartJob返回false的话,无论后台任务是否完成,JobService都将被强制销毁
  • onStartJob里执行耗时操作的话,如果操作了UI会导致ANR,如果不操作UI,当耗时逻辑(8s以上)结束之后,该JobService也会被强制停止和销毁
  • onStartJob如果新建工作线程执行后台逻辑,是存在一个10min的限制,10min后即便后台任务还没结束,JobService也会被强行停止和销毁

后台需要执行简单的任务
onStartJob可以直接返回false,告诉JobScheduler可以立即销毁Job,比如:发送IDLE状态变化的广播

后台执行耗时操作
onStartJob中新建工作线程执行耗时逻辑,并且返回true,这个时候,JobScheduler不会销毁Job,会有10min的等待时间,后台任务完成后应该自行调用jobFinished来通知JobScheduler可以立即销毁Job了,比如:网络请求获取数据

后台执行无法预估处理时间的耗时任务
这种情况就比较复杂了,除了需要在onStartJob启动工作线程后台执行逻辑,并且返回为true以外,建议还需要在onStopJob里加入以下逻辑

  1. 结束后台线程,回收资源**
  2. 保存本次任务的状态和临时文件
  3. 返回true,让系统再次启动Job
  4. 当任务重启后,读取上次任务的状态和临时文件继续完成未完成的Job
    比如:不确定大小的下载任务

二、JobService原理:

2.1 JobSchedulerService 初始化:

init:
这里写图片描述
constructor:

Constructor:
784    public JobSchedulerService(Context context) {
785        super(context);
786        //MIUI MOD:
787        //mHandler = new JobHandler(context.getMainLooper());
788        mHandler = new JobHandler(com.android.server.MiuiFgThread.get().getLooper());
791        mJobs = JobStore.initAndGet(this);
792
793        // Create the controllers.
794        mControllers = new ArrayList<StateController>();
795        mControllers.add(ConnectivityController.get(this));
797        ...
802    }

onStart()

onStart()
805    public void onStart() {
806        publishLocalService(JobSchedulerInternal.class, new LocalService());
807        publishBinderService(Context.JOB_SCHEDULER_SERVICE, mJobSchedulerStub);
808    }

onBootPhase()

@Override
811    public void onBootPhase(int phase) {
812        if (PHASE_SYSTEM_SERVICES_READY == phase) {
834        } else if (phase == PHASE_THIRD_PARTY_APPS_CAN_START) {
835            synchronized (mLock) {
851                mJobs.forEachJob(new JobStatusFunctor() {
852                    @Override
853                    public void process(JobStatus job) {
854                        for (int controller = 0; controller < mControllers.size(); controller++) {
855                            final StateController sc = mControllers.get(controller);
856                            sc.maybeStartTrackingJobLocked(job, null);
857                        }
858                    }
859                });
863            }
865        } else if (phase == PHASE_BOOT_COMPLETED) {
866            mHandler.obtainMessage(MSG_CHECK_JOB).sendToTarget();
868        }
869    }

JobSchedulerService的初始化流程:
这里写图片描述

2.2 JobSchedulerService家族大体框架:

这里写图片描述

2.2.1 JobInfo JobStatus
2.2.2 JobStore
2.2.3 StateController

这里写图片描述

TimeController举例:
这里写图片描述

job.isReady():

JobStatus.java
public boolean isReady() {final boolean deadlineSatisfied = (!job.isPeriodic() && hasDeadlineConstraint()&& (satisfiedConstraints & CONSTRAINT_DEADLINE) != 0);final boolean notIdle = (satisfiedConstraints & CONSTRAINT_APP_NOT_IDLE) != 0;final boolean notDozing = (satisfiedConstraints & CONSTRAINT_DEVICE_NOT_DOZING) != 0|| (job.getFlags() & JobInfo.FLAG_WILL_BE_FOREGROUND) != 0;return (isConstraintsSatisfied() || deadlineSatisfied) && notIdle && notDozing;
}

这篇关于Android Fk: JobService的使用解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/825596

相关文章

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

ModelMapper基本使用和常见场景示例详解

《ModelMapper基本使用和常见场景示例详解》ModelMapper是Java对象映射库,支持自动映射、自定义规则、集合转换及高级配置(如匹配策略、转换器),可集成SpringBoot,减少样板... 目录1. 添加依赖2. 基本用法示例:简单对象映射3. 自定义映射规则4. 集合映射5. 高级配置匹