Android14前台服务适配指南

Android14前台服务适配指南
Android 10引入了android:foregroundServiceType属性,用于帮助开发者更有目的地定义前台服务。这个属性在Android 14中被强制要求,必须指定适当的前台服务类型。以下是可选择的前台服务类型:
camera: 相机应用。connectedDevice: 与连接的设备相关的应用。dataSync: 数据同步应用。health: 与健康相关的应用。location: 位置相关的应用。mediaPlayback: 媒体播放应用。mediaProjection: 媒体投影应用。microphone: 麦克风相关的应用。phoneCall: 电话呼叫应用。remoteMessaging: 远程消息应用。shortService: 短期服务应用。specialUse: 特殊用途应用。systemExempted: 系统例外应用。
如果应用的前台服务与上述类型无关,建议迁移到使用WorkManager或用户触发的数据传输作业等其他方式。
值得注意的是,Android 14中新增了health、remoteMessaging、shortService、specialUse和systemExempted类型。应用在清单文件中必须声明前台服务类型,如下所示:
<manifest ...><uses-permission android:name="android.permission.FOREGROUND_SERVICE" /><uses-permission android:name="android.permission.FOREGROUND_SERVICE_MEDIA_PLAYBACK" /><application ...><serviceandroid:name=".MyMediaPlaybackService"android:foregroundServiceType="mediaPlayback"android:exported="false"></service></application>
</manifest>
对于以Android 14为目标平台的应用,如果没有定义前台服务类型,系统将在调用startForeground()时引发MissingForegroundServiceTypeException异常。这一变化旨在提高应用的安全性和用户隐私保护。
Android14前台服务适配
-
声明新权限:以Android 14为目标平台的应用使用前台服务类型时,必须根据前台服务类型声明特定的权限。这些权限列在每种前台服务类型的预期用例和强制执行部分中,并被标记为“您必须在清单文件中声明的权限”。这些权限都是一般权限,无法被用户撤销。如果应用调用
startForeground()但未声明适当的前台服务类型权限,系统将引发SecurityException异常。 -
运行时包含前台服务类型:对于启动前台服务的应用,建议使用
startForeground()的重载版本,可以在其中传递一个或多个前台服务类型的值。通常,应该只声明与特定用例相关的前台服务类型。如果某个前台服务以多个类型启动,应该遵守所有类型的强制执行要求。 -
系统运行时检查:系统会检查前台服务类型的使用是否合适,并验证应用是否已请求适当的运行时权限或使用所需的API。例如,应用使用
FOREGROUND_SERVICE_TYPE_LOCATION前台服务类型时,需要请求ACCESS_COARSE_LOCATION或ACCESS_FINE_LOCATION权限。应用必须严格遵循请求权限和启动前台服务的特定顺序。在调用startForeground()之前,必须先请求并获得所需的权限。如果应用不满足前台服务的运行时要求,在调用startForeground()后,系统将抛出SecurityException,这可以防止前台服务启动,可能导致应用崩溃。
每种前台服务类型的预期用例和强制执行部分提供了有关平台强制执行要求的详细信息,应用开发者应按照这些规定来使用前台服务类型以确保应用的正常运行。
每种前台服务类型的预期用例和强制执行
仅就摄像头、连接的设备、数据同步等几项服务,说明前台服务用法,更多类型使用请参考下面链接。
https://developer.android.google.cn/about/versions/14/changes/fgs-types-required?hl=zh-cn
要使用特定的前台服务类型,应满足以下条件:
-
在清单文件中声明特定权限。
-
满足特定的运行时要求。
-
应用必须满足该类型的其中一组预期用例。
具体前台服务类型的要求如下:
-
摄像头:
- 声明权限:
FOREGROUND_SERVICE_CAMERA。 - 运行时要求:请求相机运行时权限。
- 用例:继续在后台访问相机,如支持多任务的视频聊天应用。
- 声明权限:
-
连接的设备:
- 声明权限:
FOREGROUND_SERVICE_CONNECTED_DEVICE。 - 运行时要求:必须满足特定权限或运行时权限。
- 用例:与需要蓝牙、NFC、IR、USB或网络连接的外部设备进行互动。用例可以包括蓝牙连接、网络连接等。
- 声明权限:
-
数据同步:
- 声明权限:
FOREGROUND_SERVICE_DATA_SYNC。 - 运行时要求:无。
- 用例:数据传输操作,如数据上传、备份、导入导出、文件处理等。这个前台服务类型将被废弃,建议使用WorkManager或用户发起的数据传输作业。
- 声明权限:
-
健康(预览版,Android 14新增):
- 声明权限:
FOREGROUND_SERVICE_HEALTH。 - 运行时要求:必须满足特定权限或运行时权限。
- 用例:为健身类别的应用提供支持的长时间运行用例,如健身追踪器。
- 声明权限:
-
位置:
- 声明权限:
FOREGROUND_SERVICE_LOCATION。 - 运行时要求:请求位置信息使用权限,包括
ACCESS_COARSE_LOCATION或ACCESS_FINE_LOCATION。 - 用例:长时间运行的位置信息用例,如导航和位置信息分享。
- 声明权限:
满足这些前台服务类型的要求是确保应用能够正常使用相关功能的关键。开发者应在清单文件和运行时权限请求中遵循这些规定,以确保应用能够以前台服务类型的方式正常运行。
前台服务用于用户数据传输业务
在 Android 14 中,前台服务规则更严格,要求应用满足条件才能使用前台服务。引入新 API 指定用户发起的数据传输作业,适用于长时间数据传输。这类作业需要用户手动启动,具备 RUN_USER_INITIATED_JOBS 权限。改进旨在提高系统稳定性和用户体验。
开启用户作业
要运行用户发起的作业,请执行以下步骤:
- 在清单文件中声明
RUN_USER_INITIATED_JOBS权限:
<manifest ...><uses-permission android:name="android.permission.RUN_USER_INITIATED_JOBS" /><application ...>...</application>
</manifest>
- 在构建
JobInfo对象时,使用新的setUserInitiated()和setDataTransfer()方法。建议您提供预估的载荷大小,可以使用setEstimatedNetworkBytes()方法:
val networkRequestBuilder = NetworkRequest.Builder().addCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET).addCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED)val jobInfo = JobInfo.Builder()// ....setUserInitiated(true).setDataTransfer(true).setRequiredNetwork(networkRequestBuilder.build()).setEstimatedNetworkBytes(1024 * 1024 * 1024)// ....build()
- 在应用可见时或满足允许的条件时调度作业:
val jobScheduler: JobScheduler =context.getSystemService(Context.JOB_SCHEDULER_SERVICE) as JobScheduler
jobScheduler.schedule(jobInfo)
- 执行作业时,请确保为
JobService对象调用setNotification(),用于通知用户作业正在运行:
val notification = Notification.Builder(applicationContext, NOTIFICATION_CHANNEL_ID).setContentTitle("My user-initiated data transfer job").setSmallIcon(android.R.mipmap.myicon).setContentText("Job is running").build()class CustomJobService : JobService() {override fun onStartJob(params: JobParameters?): Boolean {setNotification(params, notification.id, notification, JobService.JOB_END_NOTIFICATION_POLICY_DETACH)// 执行作业任务。}
}
-
请注意,如果未及时调用
setNotification(),可能导致应用出现 ANR。 -
定期更新通知,以便用户了解作业的状态和进度。如果在安排作业之前无法确定传输大小,请在了解传输大小后使用新的 API
updateEstimatedNetworkBytes()更新传输大小。 -
在作业执行完成后,调用
jobFinished()以向系统表明作业已完成,或者需要重新调度作业。
停止用户作业
-
停止用户发起的数据传输作业:
- 用户和系统都可以停止用户发起的传输作业。
- 用户可以通过任务管理器中的 Stop 操作来停止用户发起的传输作业。
- 当用户停止作业时,系统会终止应用的进程,包括所有正在运行的作业或前台服务。
- 系统不会调用
onStopJob()来终止作业,也会阻止重新调度用户可见的作业。 - 建议在发布的作业通知中提供控件,以方便用户停止和重新调度作业。
- 在某些情况下,任务管理器中可能不会显示 Stop 按钮或作业根本不会出现。
-
系统提供的停止作业:
- 与常规作业不同,用户发起的数据传输作业不受应用待机模式存储分区配额的影响。
- 但是,系统仍会停止作业,原因包括:
- 不再满足开发者定义的约束条件。
- 系统认为该作业的运行时间超出了完成数据传输所需的时间。
- 系统需要优先考虑系统运行状况,可能由于设备温度过高而停止作业。
- 应用进程因设备内存不足而被终止。
- 当系统停止作业时(不是因内存不足),系统会调用
onStopJob(),并在系统认为最佳的时间重试作业。 - 开发者应确保应用能够保留数据传输状态,即使未调用
onStopJob(),并在再次调用onStartJob()时能够恢复状态。
用户作业约束条件
总结如下:
-
为了支持在最佳时间点运行的作业,Android 提供了能够为每种作业类型分配约束条件的功能,这一功能从 Android 13 开始可用。
-
用户发起的数据传输作业允许使用的约束条件包括:
setBackoffCriteria(JobInfo.BACKOFF_POLICY_EXPONENTIAL)setClipData()setEstimatedNetworkBytes()setMinimumNetworkChunkBytes()setPersisted()setNamespace()setRequiredNetwork()setRequiredNetworkType()setRequiresBatteryNotLow()setRequiresCharging()setRequiresStorageNotLow()
测试
测试应用作业的步骤包括:
- 获取作业 ID。
- 通过命令
adb shell cmd jobscheduler run -f APP_PACKAGE_NAME JOB_ID可以立即运行或重试已停止的作业。 - 通过命令
adb shell cmd jobscheduler timeout TEST_APP_PACKAGE TEST_JOB_ID可以模拟系统强行停止作业,例如因系统运行状况或超出配额条件。
相关文章:
Android14前台服务适配指南
Android14前台服务适配指南 Android 10引入了android:foregroundServiceType属性,用于帮助开发者更有目的地定义前台服务。这个属性在Android 14中被强制要求,必须指定适当的前台服务类型。以下是可选择的前台服务类型: camera: 相机应用。…...
Spring Boot中使用Spring Data JPA访问MySQL
Spring Data JPA是Spring框架提供的用于简化JPA(Java Persistence API)开发的数据访问层框架。它通过提供一组便捷的API和工具,简化了对JPA数据访问的操作,同时也提供了一些额外的功能,比如动态查询、分页、排序等。 …...
Go 语言函数闭包(匿名函数)
Go 语言函数闭包(匿名函数) 在Go语言中,闭包是一种特殊的匿名函数,它可以捕获并访问其周围的变量。闭包允许将函数与其引用的环境捆绑在一起,使得函数可以在其创建的范围之外继续使用这些变量。以下是关于Go语言闭包的…...
2023年11月编程语言流行度排名
点击查看最新编程语言流行度排名(每月更新) 2023年11月编程语言流行度排名 编程语言流行度排名是通过分析在谷歌上搜索语言教程的频率而创建的 一门语言教程被搜索的次数越多,大家就会认为该语言越受欢迎。这是一个领先指标。原始数据来自…...
apache-maven-3.6.3 安装配置教程
链接:https://pan.baidu.com/s/1RkMXipnvac9EKcZyUStfGQ?pwdl32m 提取码:l32m 1. 将 maven 压缩包解压至指定文件夹 2. 配置环境变量 (1)打开此电脑-> 鼠标右键选择属性->点击高级系统设置 (2)点…...
你一般什么时候使用GPT
一般在寻求帮助的时候才使用gpt 一个优秀的gpt项目gpt-on-web...
kubernetes (k8s)的使用
一、kubernetes 简介 谷歌2014年开源的管理工具项目,简化微服务的开发和部署。 提供功能:自愈和自动伸缩、调度和发布、调用链监控、配置管理、Metrics监控、日志监控、弹性和容错、API管理、服务安全等。官网:https://kubernetes.io/zh-cn…...
RK3568平台开发系列讲解(音视频篇)RTMP 推流
🚀返回专栏总目录 文章目录 一、RTMP 的工作原理二、RTMP 流媒体服务框架2.1、Nginx 流媒体服务器2.2、FFmpeg 推流沉淀、分享、成长,让自己和他人都能有所收获!😄 📢目前常见的视频监控和视频直播都是使用了 RTMP、RTSP、HLS、MPEG-DASH、 WebRTC流媒体传输协议等。 R…...
掌握这几个技巧,才敢称为Jenkins大神!
01、Performance插件兼容性问题 自由风格项目中,有使用 Performance 插件收集构建产物,但是截至到目前最新版本(Jenkins v2.298,Performance:v3.19),此插件和Jenkins都存在有兼容性问题…...
帷幄内容管理系统:从立人设、做内容到定向投流,品牌 KOS 体系打造「百万导购」
随着公域流量越来越贵,获客成本越来越高,品牌们已经越来越不满足于高曝光,而是更多地关注起销售转化率。继 KOL、KOC(关键意见消费者) 之后,KOS(关键意见销售)营销模式走入品牌的视野…...
5.vue3项目(五):实现顶部导航栏功能:导航栏静态搭建,菜单折叠功能实现,面包屑动态展示路径,刷新页面功能,全屏功能
目录 一、左侧菜单栏刷新,不要合并菜单 二、顶部tabbar静态搭建 1.新建文件 2.编辑页面 3.结果测试...
unittest 统计测试执行case总数,成功数量,失败数量,输出至文件,生成一个简易的html报告带饼图
这是一个Python的单元测试框架的示例代码,主要用于执行测试用例并生成测试报告。其中,通过unittest模块创建主测试类MainTestCase,并加载其他文件中的测试用例,统计用例的执行结果并将结果写入文件,最后生成一个简单的…...
推荐一款功能强大的在线文件预览工具-kkFileView
程序员的公众号:源1024,获取更多资料,无加密无套路! 最近整理了一波电子书籍资料,包含《Effective Java中文版 第2版》《深入JAVA虚拟机》,《重构改善既有代码设计》,《MySQL高性能-第3版》&…...
【Linux】磁盘阵列,了解不同raid的特点
一、raid和阵列卡介绍 1、什么是磁盘阵列: 磁盘阵列是利用虚拟化存储技术把很多块独立的磁盘组合成一个容量巨大的磁盘组,利用个别磁盘提供数据所产生加成效果提升整个磁盘系统效能。利用这项技术,将数据切割成许多区段,分别存放…...
Go 语言初探:从基础到实战
1.Go概述 程序是一段计算机指令的有序组合。程序算法数据结构。任何程序都可以将模块通过三种基本的控制结构(顺序、分支、循环)进行组合来实现。 Go(也称为Golang)是一种由Google开发的开源编程语言。设计目标是使编程更简单、…...
Kotlin文件和类为什么不是一对一关系
在Java中,一个类文件的public类名必须和文件名一致,如何不一致就会报异常,但是在kotlin的文件可以和类名一致,也可以不一致。这种特性,就跟c有点像,毕竟c的.h 和 .cpp文件是分开的。只要最终编译的时候对的…...
Kubernetes实战(四)-部署docker harbor私有仓库
1 Docker原生私有仓库Registry 1.1 原生私有仓库Registry概述 Docker的仓库主要分两类: 私有仓库公有仓库 共有仓库只要在官方注册用户,登录即可使用。但对于仓库的使用,企业还是会有自己的专属镜像,所以私有库的搭建也是很有…...
IDEA JAVA项目 导入JAR包,打JAR包 和 JAVA运行JAR命令提示没有主清单属性
一、导入JAR包 1、java项目在没有导入该jar包之前,如图:2、点击 File -> Project Structure(快捷键 Ctrl Alt Shift s),点击Project Structure界面左侧的“Modules”如图:3.在 “Dependencies” 标签…...
c#输入和输出
第一个c#程序 /* c#在编译时首先编译为通用中间语言(IL代码),并且将其存在程序集中 c#的程序集包括扩展名为.exe的可执行文件和扩展名为.dll的可供其他程序调用的库文件c#在执行时首先将程序集加载到CLR中,然后通过即时编译器编译…...
设置区块链节点输出等级为警告级,并把日志存储阈值位100MB并验证;
题目 获取指定区块链节点输出等级为警告级,并设置日志存储阈值位100MB并验证; 操作步骤 1.切换目录 cd nodes/127.0.0.1/node0 2.打开配置文件并修改 vim config.ini warn:警告...
手游刚开服就被攻击怎么办?如何防御DDoS?
开服初期是手游最脆弱的阶段,极易成为DDoS攻击的目标。一旦遭遇攻击,可能导致服务器瘫痪、玩家流失,甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案,帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...
高频面试之3Zookeeper
高频面试之3Zookeeper 文章目录 高频面试之3Zookeeper3.1 常用命令3.2 选举机制3.3 Zookeeper符合法则中哪两个?3.4 Zookeeper脑裂3.5 Zookeeper用来干嘛了 3.1 常用命令 ls、get、create、delete、deleteall3.2 选举机制 半数机制(过半机制࿰…...
c#开发AI模型对话
AI模型 前面已经介绍了一般AI模型本地部署,直接调用现成的模型数据。这里主要讲述讲接口集成到我们自己的程序中使用方式。 微软提供了ML.NET来开发和使用AI模型,但是目前国内可能使用不多,至少实践例子很少看见。开发训练模型就不介绍了&am…...
Element Plus 表单(el-form)中关于正整数输入的校验规则
目录 1 单个正整数输入1.1 模板1.2 校验规则 2 两个正整数输入(联动)2.1 模板2.2 校验规则2.3 CSS 1 单个正整数输入 1.1 模板 <el-formref"formRef":model"formData":rules"formRules"label-width"150px"…...
稳定币的深度剖析与展望
一、引言 在当今数字化浪潮席卷全球的时代,加密货币作为一种新兴的金融现象,正以前所未有的速度改变着我们对传统货币和金融体系的认知。然而,加密货币市场的高度波动性却成为了其广泛应用和普及的一大障碍。在这样的背景下,稳定…...
比较数据迁移后MySQL数据库和OceanBase数据仓库中的表
设计一个MySQL数据库和OceanBase数据仓库的表数据比较的详细程序流程,两张表是相同的结构,都有整型主键id字段,需要每次从数据库分批取得2000条数据,用于比较,比较操作的同时可以再取2000条数据,等上一次比较完成之后,开始比较,直到比较完所有的数据。比较操作需要比较…...
淘宝扭蛋机小程序系统开发:打造互动性强的购物平台
淘宝扭蛋机小程序系统的开发,旨在打造一个互动性强的购物平台,让用户在购物的同时,能够享受到更多的乐趣和惊喜。 淘宝扭蛋机小程序系统拥有丰富的互动功能。用户可以通过虚拟摇杆操作扭蛋机,实现旋转、抽拉等动作,增…...
在 Spring Boot 项目里,MYSQL中json类型字段使用
前言: 因为程序特殊需求导致,需要mysql数据库存储json类型数据,因此记录一下使用流程 1.java实体中新增字段 private List<User> users 2.增加mybatis-plus注解 TableField(typeHandler FastjsonTypeHandler.class) private Lis…...
实战三:开发网页端界面完成黑白视频转为彩色视频
一、需求描述 设计一个简单的视频上色应用,用户可以通过网页界面上传黑白视频,系统会自动将其转换为彩色视频。整个过程对用户来说非常简单直观,不需要了解技术细节。 效果图 二、实现思路 总体思路: 用户通过Gradio界面上…...
全面解析数据库:从基础概念到前沿应用
在数字化时代,数据已成为企业和社会发展的核心资产,而数据库作为存储、管理和处理数据的关键工具,在各个领域发挥着举足轻重的作用。从电商平台的商品信息管理,到社交网络的用户数据存储,再到金融行业的交易记录处理&a…...
