Android 15 适配整理——实践版

news2025/1/6 17:35:37

背景

谷歌发布Android 15后,国内的手机厂商迅速行动,开始了新系统的适配工作。小米、OPPO、vivo和联想等金标联盟成员联合发布了适配公告,督促APP开发者在2024年8月31日前完成适配工作,否则将面临搜索标签提示、应用降级、分机型屏蔽以及应用下架等处罚措施。

可能存在的影响面有哪些

  1. UI相关:主要页面状态栏是否正常和底部是否有遮挡。包括:原生页面;h5页面 ;flutter页面。(沉浸式和非沉浸式)
  2. 通知栏样式:极光推送是否通知栏样式是否正常。
  3. 存储相关:文件下载功能:版本更新功能是否正常。图库选择图片和视频是否正常。图片资源保持功能是否正常。
  4. 权限相关:机型测试权限是否有异常。
  5. 机型覆盖:安卓15 需要兼容 OPPO、VIVO、小米等。(登录官网-找云真机测试)

Android 15云真机

因为是提前做好适配,因此,目前市面上没有android 15 系统的手机,需要申请云真机,比如,小米、oppo、vivo等。
使用云真机过程中的一些经验之谈:

  • 小米的云真机调试限制:1天8小时,每半小时需要重连一次;重连失败,可能是因为云真机被占用了。
  • vivo的云真机调试:
    • 使用国内的云真机比较正常;国外的云真机,会一直系统白屏(不确定是否浏览器问题,试了Google和360浏览器都是白屏)。
    • 如果隔段时间没用,大约半小时左右,会存在死机现象;刷新后,需要重新安装软件。
  • 使用体感:vivo的云真机比小米云真机要流畅一点;但是vivo放置久了,长时间无输入操作,也会存在卡顿现象。
  • 云真机链接:
  • vivo云真机
  • 小米云真机

适配方案流程:

前提:在做本次Android15适配之前,项目已经做好Android 11 以上的适配工作。主要是:存储、权限、第三方sdk适配、隐私权益这些流程的妥善处理。可以看之前整理的另一篇文章:《Android 11 适配——整理总结篇》

影响Android 15上所有应用的行为变更

处理最低sdk适配
  • 最低可安装TargetSDK级别为24!
    如果尝试安装一个针对较低API级别的应用程序会导致安装失败,并在Logcat中出现以下消息:
INSTALL_FAILED_DEPRECATED_SDK_VERSION: App package must target at least SDK version 24, but found 7
应用 STOPPED 状态的变化

当应用进入 STOPPED 状态时,系统会自动取消应用的所有 PendingIntent。这意味着任何未决的操作,包括但不限于定时任务和远程服务调用,都将被中断。
在此状态下,应用的小部件(widgets)也会被禁用,表现为灰色不可交互状态,直到应用再次启动。
用户通过直接或间接操作将应用从 STOPPED 状态唤醒时,系统会发送 ACTION_BOOT_COMPLETED 广播给应用。这允许应用重新注册其 PendingIntent 和恢复其他必要的后台活动。
应用可以通过 ApplicationStartInfo.wasForceStopped() 接口来检测自身是否曾处于 STOPPED 状态,以便采取相应的恢复措施。
PendingIntent
PendingIntent 则是对 Intent 的一种封装,它提供了在未来的某个不确定时间点执行 Intent 的能力,并且可以在不同的进程之间安全地传递。PendingIntent 主要有以下用途:
异步执行:允许在未来的某个时刻执行 Intent,即使是在应用不处于前台的情况下。
跨进程调用:可以安全地将 Intent 传递给其他应用或进程,因为 PendingIntent 保证了 Intent 的执行上下文和权限。
延迟绑定:在创建 PendingIntent 时不需要立即知道所有参数,可以在稍后的某个时间点再进行绑定和执行。

影响以Android 15为目标平台应用的行为变更

新的媒体处理前台服务类型

项目中,没有对应场景需要用到该新的服务类型

Android 15引入了一种新的前台服务类型,即mediaProcessing。
该系统允许一个应用的媒体处理服务在24小时内运行总共6个小时,之后系统会调用正在运行的服务的Service.onTimeout(int, int)方法(在Android 15中引入)。
要使用这种前台服务类型,需要在manifest中申明相关权限和foregroundServiceType属性:
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools">
    <uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
    <uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
    <uses-permission android:name="android.permission.FOREGROUND_SERVICE_MEDIA_PROCESSING" />
.....
        <service
            android:name=".MyService"
            android:enabled="true"
            android:exported="true"
            android:foregroundServiceType="mediaProcessing"
            >
    </service>
......

Receiver 启动前台服务的限制

BOOT_COMPLETED 广播接收器启动前台服务有了新的限制,不得启动以下类型的前台服务:dataSync、camera、mediaPlayback、phoneCall、mediaProjection、microphonemicrophone(此限制自 Android 14 起生效)
如果BOOT_COMPLETED接收器尝试启动任何类型的前台服务,系统将抛出异常:ForegroundServiceStartNotAllowedException。

IntentFilter规范

在 Android 15 中,系统对 Intent 的安全性和精确性进行了显著增强,主要体现在两个关键方面:
精确匹配 Intent-Filters:当一个应用尝试通过 Intent 启动另一个应用中的组件(如 Activity)时,发送的 Intent 必须严格符合接收组件声明的 intent-filters。这意味着,Intent 中的属性(如 category、data 和 action)必须与接收方在清单文件(AndroidManifest.xml)中定义的 intent-filters 完全匹配,才能成功启动目标组件。这一改变确保了组件只能被预期的 Intent 所激活,增强了跨应用交互的安全性。
强制 Intent 包含 Action:所有用于启动 Activity 或 Service 的 Intent 现在都必须包含一个明确的 action 字段。如果没有指定 action,该 Intent 将不会匹配到任何 intent-filters,从而无法启动任何组件。这一要求进一步提高了 Intent 的意图清晰度和安全性,防止了因缺少行动指令而可能引发的意外行为或安全漏洞。
通过启动严格模式可以发现应用中潜在问题

fun onCreate() {
    StrictMode.setVmPolicy(VmPolicy.Builder()
        .detectUnsafeIntentLaunch()
        .build()
    )
}
Edge-to-edge 模式

概述
Edge-to-edge模式是Android 15引入的一种布局策略,旨在使应用界面充分利用整个屏幕空间,包括状态栏和导航栏区域。在该模式下,应用界面将扩展至屏幕边缘,且无法自定义状态栏背景颜色,导航栏的背景透明度默认设为80%。
特性
默认启用:针对Android 15设备,所有targetSdkVersion为35或更高的应用将默认启用Edge-to-edge模式。
布局影响:启用该模式可能会影响应用的布局设计,需要开发者进行相应的适配工作。
限制条件:在Edge-to-edge模式下,非浮动窗口的layoutInDisplayCutoutMode属性必须设置为LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS,否则会抛出IllegalArgumentException。
API弃用:在Android 15平台上,之前用于设置系统栏颜色的API如setStatusBarColor和setNavigationBarColor将被弃用,即使调用,系统也会保持默认的沉浸式体验。
应用适配
targetSdkVersion >= 35:应用将强制进行全屏展示,状态栏和导航栏保持透明。若应用布局需避开状态栏和导航栏,应确保在XML布局文件中设置:

android:fitsSystemWindows="true"

targetSdkVersion < 35:应用默认不会启用Edge-to-edge特性,维持原有状态栏和导航栏的处理方式。
注意事项
开发者需检查应用的布局文件,确保在targetSdkVersion >= 35的应用中,所有需要避开状态栏和导航栏的布局元素都正确设置了fitsSystemWindows属性。
对于需要自定义状态栏和导航栏颜色或透明度的应用,应考虑使用其他方法替代已弃用的API,例如使用主题和样式资源来控制UI的外观

其他变更:

  • 达到资源限制时,直接和分流音频播放会使之前打开的直接或分流音轨失效
  • 支持16KB Page Size
  • 默认开启预测性返回动画
  • 对启动前台服务的 BOOT_COMPLETED 广播接收器的限制
  • 对请求音频焦点的限制
  • ……

觉得不错的新功能

  • 提供了ApplicationStartInfo API供开发者获取应用启动的相关信息。
getDefiningUid(): 大多数情况下与通常认为的UID相同,使用了android:useAppZygote属性和Context.BIND_EXTERNAL_SERVICE标志位的service可能会导致这个字段有所不同。
getIntent(): 就是引发这个应用启动的intent。
getLaunchMode(): 启动模式,本例中为0,意为LAUNCH_MODE_STANDARD。
getPackageUid(): 即App安装时分配到的UID。
getPid()、getProcessName(): 即进程Pid、进程名。
getRealUid(): 大多数情况下与PackageUid相同,在涉及应用分身、多用户等情况下可能会不同。
getReason(): 应用启动的原因。可能返回的值有:
  0,START_REASON_ALARM,应用被闹钟启动;
  1, START_REASON_BACKUP,因为执行backup而启动;
  2,START_REASON_BOOT_COMPLETE,应用因为开机完成启动;

getStartType(): App自动的类型,比如:
  1,START_TYPE_COLD,冷启动;
  3,START_TYPE_HOT,热启动

getStartupState(): 启动状态。如:
  0,STARTUP_STATE_STARTED,启动成功;
  1,STARTUP_STATE_ERROR,启动失败;
等等。

getStartupTimestamps(): 得到不同启动阶段的时间戳,比如:START_TIMESTAMP_LAUNCH,START_TIMESTAMP_BIND_APPLICATION等等。

wasForceStopped(): 应用是否是被forcestop结束的。开发者可以利用这个API决定应用启动后是否重新注册闹钟、JobScheduler等被forcestop清除掉的机制。
开发者也可以使用addApplicationStartInfoCompletionListener与removeApplicationStartInfoCompletionListener方法来添加和删除监听器,监听器会在应用启动信息生成完毕后回调
  • 照片选择改进:当 App 得到部分媒体权限时,应用可以仅突出显示最近选择的照片和视频,这可以改善频繁请求访问照片和视频的用户体验,详细可以通过 ContentResolver 在 MediaStore 启用时查询 QUERY_ARG_LATEST_SELECTION_ONLY 参数来实现。
  • 局部屏幕共享
  • SQLite 数据库改进: SQLite 性能最佳实践
  • ……

关于是否将 targetSdkVersion 设置为Android 15 的思考

  • 结论:当前版本30,待15稳定后再设置为目标版本。
  • 原因:
    1. 目前查看Android 15 官方更新日志,在2024年7月18日 又将本应该 Android 15 对应的api 34 调整为 api 35了。官方表明,仍在积极开发阶段,可能存在稳定性问题。

在这里插入图片描述
3. 目前我们的适配、上线日期都是在正式版之前的,目标版本不应该以不够稳定的15 beta为目标版本,而应该以稳定版本为主。
4. 目前国内最低版本要求是不低于24;如果上线Google应用商店,最低目标版本是不低于29;目前项目的最低版本已声明为24(原21),目标版本已声明为30。满足当前应用市场要求,并最大程度确保了版本稳定性。
5. vivo平台确认截图:
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1948180.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

MySQL数据库-库表操作

一、SQL语句基础 1.SQL简介 SQL&#xff1a;结构化查询语言&#xff08;Structured Query Language&#xff09;&#xff0c;在关系型数据库上执行数据操作、数据检索以及数据维护的标准语言。使用SQL语句&#xff0c;程序员和数据库管理员可以完成如下任务&#xff1a; &am…

Leetcode—426. 将二叉搜索树转化为排序的双向链表【中等】Plus

2024每日刷题&#xff08;148&#xff09; Leetcode—426. 将二叉搜索树转化为排序的双向链表 实现代码 /* // Definition for a Node. class Node { public:int val;Node* left;Node* right;Node() {}Node(int _val) {val _val;left NULL;right NULL;}Node(int _val, Nod…

开放式耳机怎么选购性价比高?五大好评不断的爆款分享

长期使用入耳式耳机的朋友应该都会有这种担忧&#xff0c;经常塞着耳朵听歌&#xff0c;耳机上的细菌得不到及时的清理&#xff0c;不够健康卫生&#xff0c;还有诱发耳部发炎感染的风险。而开放式耳机的出现恰好缓解了这种担忧。如果你也想知道开放式耳机是否真有网上宣传的那…

Photos框架 - 自定义媒体选择器(UI列表)

引言Photos框架 - 自定义媒体资源选择器&#xff08;数据部分&#xff09;-CSDN博客 关于自定义媒体选择器上一篇博客我们已经介绍了使用Photos获取媒体资源数据和处理媒体资源数据&#xff0c;有了数据&#xff0c;UI的实现就比较灵活了&#xff0c;我就以上面的设计样式为例…

学习React(描述 UI)

React 是一个用于构建用户界面&#xff08;UI&#xff09;的 JavaScript 库&#xff0c;用户界面由按钮、文本和图像等小单元内容构建而成。React 帮助你把它们组合成可重用、可嵌套的 组件。从 web 端网站到移动端应用&#xff0c;屏幕上的所有内容都可以被分解成组件。在本章…

08 字符串和字节串

使用单引号、双引号、三单引号、三双引号作为定界符&#xff08;delimiter&#xff09;来表示字符串&#xff0c;并且不同的定界符之间可以相互嵌套。 很多内置函数和标准库对象也都支持对字符串的操作。 x hello world y Python is a great language z Tom said, "Le…

数据结构----算法复杂度

1.数据结构前言 数据是杂乱无章的&#xff0c;我们要借助结构将数据管理起来 1.1 数据结构 数据结构(Data Structure)是计算机存储、组织数据的⽅式&#xff0c;指相互之间存在⼀种或多种特定关系的数 据元素的集合。没有⼀种单⼀的数据结构对所有⽤途都有⽤&#xff0c;所…

【Python机器学习】k-近邻算法简单实践——电影分类

k-近邻算法&#xff08;KNN&#xff09;的工作原理是&#xff1a;存在一个样本数据集合&#xff0c;也被称为训练样本集&#xff0c;并且样本集中每个数据都存在标签&#xff0c;即我们知道样本集中每一数据与所属分类的对应关系&#xff0c;输入没有标签的数据后&#xff0c;将…

内存管理概念 (二)

目录 一 . 基本分页存储管理分页存储的几个基本概念页面 与 页面大小地址结构页表 基本地址变换机构具有快表的地址变换机构两级页表 二. 基本分段存储管理分段段表地址变换机构页表和分段的对比段的共享与保护 三. 段页式存储管理分页&#xff0c;分段管理优缺点分页分段段页式…

MATLAB中reset用法

目录 语法 说明 示例 重置坐标区和图窗属性 参数说明 局限性 reset函数的功能是重置图形对象属性。 语法 reset(h) 说明 reset(h) 将指定图形对象的所有属性重置为其默认值。没有默认值的属性不重置。 MATLAB 不会重置任何图形对象的 Position 或 Units 属性。此外&am…

Java-根据前缀-日期-数字-生成流水号(不重复)

&#x1f388;边走、边悟&#x1f388;迟早会好 小伙伴们在日常开发时可能会遇到的业务-生成流水号&#xff0c;在企业中可以说是比较常见的需求&#xff0c; 可以采用"前缀日期数字"的方式&#xff08;ps:此方式是需要用到缓存的&#xff09;前缀&#xff1a;为了…

SpringBoot集成Kaptcha验证码

Hi &#x1f44b;, Im shy 有人见尘埃&#xff0c;有人见星辰 1. 什么是Kaptcha验证码? Kaptcha是一个强大的开源Java验证码生成库,由Google开发。它能够生成高度可配置的图片验证码,主要用于防止自动化程序滥用web应用,提高应用的安全性。 2. Kaptcha的主要特性 Kaptch…

Kafka快速入门+SpringBoot简单的秒杀案例

1. 主题相关 1.1 创建主题 kafka-topics.sh --create --bootstrap-server [服务器地址] --replication-factor [副本数] --partitions [分区数] --topic [主题名]liberliber-VMware-Virtual-Platform:/home/zookeeper$ docker-compose exec kafka /bin/bash #进入kafka容器 b…

还手动抄字幕?学会这3个视频转文字方法,轻松提取视频中的字幕!

大家有尝试过考试前极限抱佛脚吗&#xff1f; 在下不才&#xff0c;曾经试过一次&#xff0c;轻松在及格线低空飘过【大家不要学不要学不要学&#xff0c;重要的事情说三遍&#xff01;&#xff01;&#xff01;】 至于我当时究竟是怎么做到的呢&#xff1f;其实这里面有点小…

走进数组的奇妙之旅(1)-学习笔记

引言&#xff1a; 在前几篇文章中&#xff0c;我们深入探讨了函数的奥秘。在讲述函数知识的过程中&#xff0c;我们邂逅了一个新的概念&#xff0c;你或许还记得在演示 strcpy函数时&#xff0c;出现的这行代码&#xff1a;char1[20]{0};。当时&#xff0c;你是否感到好奇&…

国产光电耦合器2024年的机遇与挑战

随着科技的飞速发展&#xff0c;2024年对于国产光电耦合器行业来说&#xff0c;无疑是充满机遇与挑战的一年。本文将深入探讨该行业在技术创新、市场竞争、5G时代、新兴应用领域和国际市场拓展方面的现状及未来前景。 技术创新的黄金期 物联网和人工智能技术的迅猛发展&#x…

【实在RPA案例集】实在智能助力中国烟草11省40余家多场景自动化!

近年来&#xff0c;为深入贯彻行业数字化转型战略部署和发展新质生产力体制机制&#xff0c;诸多省市烟草公司及中烟公司大力推进烟草行业数字化转型&#xff0c;然而烟草行业在数字化转型过程中始终存在一个核心痛点&#xff0c;即数据整合的复杂性、系统间的兼容性问题&#…

众人吹捧的Exo并不是真正的分布式推理,而无人问津的Cake或许才是

之前就看到不少抖音AI区的博主吹火爆外网的Exo项目支持多台苹果机运行一个 Llama 70B。我就觉得这个事情或许有些不对劲&#xff0c;而随着最近 Meta 出了 405B&#xff0c;又有朋友向我推荐Exo&#xff0c;我不禁想知道一个 Bonjour 的 Zeroconf 怎么就让 MLX 支持 P2P 的分布…

IP 泄露: 原因与避免方法

始终关注您的IP信息&#xff01; 您的IP地址不仅显示您的位置&#xff0c;它包含几乎所有的互联网活动信息&#xff01; 如果出现IP泄漏&#xff0c;几乎所有的信息都会被捕获甚至非法利用&#xff01; 那么&#xff0c;网站究竟如何追踪您的IP地址&#xff1f;您又如何有效…

【前端学习笔记】CSS基础一

一、什么是CSS 1.CSS 介绍 CSS&#xff08;Cascading Style Sheets&#xff0c;层叠样式表&#xff09;是一种用来控制网页布局和设计外观的样式语言。它使得开发者可以分离网页的内容&#xff08;HTML&#xff09;和表现形式&#xff08;样式&#xff09;&#xff0c;提高了…