Android 13 变更及适配攻略

news2025/1/12 6:08:29

在这里插入图片描述

准备工作

首先将我们项目中的 targetSdkVersioncompileSdkVersion 升至 33。

影响Android 13上所有应用

1.通知受限

对新安装的应用的影响:

如果用户在搭载 Android 13 或更高版本的设备上安装您的应用,应用的通知默认处于关闭状态。在您请求新的权限且用户向您的应用授予该权限之前,您的应用都将无法发送通知。

  • 如果您的应用以 Android 13 或更高版本为目标平台,应用将可以完全自行控制权限对话框的显示时间。您可以借此机会向用户说明应用需要此权限的原因,进而鼓励他们授予该权限。
  • 如果您的应用以 12L(API 级别 32)或更低版本为目标平台,在您创建通知渠道后您的应用首次启动 activity 时,或在您的应用启动一个 activity,然后创建它的第一个通知渠道时,系统会显示该权限对话框。这通常是在应用启动时。

对现有应用更新的影响:

为了最大限度地减少与通知权限相关的中断,当用户将其设备升级到Android 13或更高版本后,系统会自动向所有符合条件的应用预先授予相应权限。换言之,这些应用可以继续向用户发送通知,而用户不会看到运行时权限提示。

适配方法:

应用以Android 13或更高版本为目标平台,需要在应用程序manifest文件中声明POST_NOTIFICATIONS权限,并完成与请求其他运行时权限类似的流程。

<manifest ...>
    <uses-permission android:name="android.permission.POST_NOTIFICATIONS"/>
</manifest>

请求权限前,可以使用areNotificationsEnabled()方法检查用户是否已启用通知。Android 13上做权限申请,Android 13一下可以引导设置页开启。

官方文档建议等到用户熟悉您的应用之后,再请求授予权限。下面列举了几个适合显示通知权限提示的时机:

  • 用户点按“提醒铃铛”按钮时。
  • 用户选择关注他人的社交媒体帐号时。
  • 用户提交外卖订单时。

下图是关于请求通知权限的建议用户驱动工作流程。仅当 shouldShowRequestPermissionRationale() 返回true时,才有必要显示中间屏幕。

在这里插入图片描述

2.intent过滤器会屏蔽不匹配的intent

当您的应用向以Android 13或更高版本为目标平台的其他应用的导出组件发送 intent 时,仅当该intent与接收应用中的<intent-filter>元素匹配时,系统才会传送该 intent。换言之,系统会屏蔽所有不匹配的intent,但以下情况除外:

  • 目标组件未声明任何<intent-filter>
  • 同一应用内发送的intent
  • 由系统发送的intent
  • 具有ROOT权限的进程发送的intent

如果接收方应用升级到Android 13或更高版本,仅当intent与其声明的 元素匹配时,源自外部应用的所有intent才会传送到导出组件,而不考虑发送应用的目标SDK版本。

所以我们需要检查应用内是否有通过Intent方式启动其他App或发送广播,同时检查actiondata等信息是否准确。

影响以Android 13或更高版本为目标平台的应用

1.GestureDetector

升到33后,发现编译失败。一看是kotlin实现的GestureDetector.OnGestureListener接口报错,提示onScroll方法的MotionEvent参数不会为空。原因是Android 13上这里的代码多了一个@NonNull的注解。

在这里插入图片描述

本以为删除可空的“?”就正常了,但发现在低版本手机上会报错:

Fatal Exception: java.lang.NullPointerException: 
Parameter specified as non-null is null: method android.view.GestureDetector.onTouchEvent, parameter e1

因为低版本手机上,这里拿到的MotionEvent为空。但是在kotlin中,变量不可空时,它会使用checkNotNullParameter方法校验,如果为null就会发生上面的异常。

在这里插入图片描述

所以如果你是用java实现的,不会存在这个问题。那么解决方法之一就是改用java实现。或者可以加上@Suppress("NOTHING_TO_OVERRIDE", "ACCIDENTAL_OVERRIDE")这样就不会生成checkNotNullParameter校验代码。

如果适配过程中有类似问题,不要盲目删除“?”。需要注意一下低版本的运行情况。

2.读取媒体文件权限

对于目标版本为Android 13的应用,细化了READ_EXTERNAL_STORAGE权限,使用READ_MEDIA_IMAGEREAD_MEDIA_VIDEOREAD_MEDIA_AUDIO替代READ_EXTERNAL_STORAGE

如果用户之前向您的应用授予了 READ_EXTERNAL_STORAGE 权限,系统会自动向您的应用授予细化的这三个媒体权限。

适配说明:

<manifest ...>
    <!-- 按需添加 -->
    <uses-permission android:name="android.permission.READ_MEDIA_IMAGES" />
    <uses-permission android:name="android.permission.READ_MEDIA_AUDIO" />
    <uses-permission android:name="android.permission.READ_MEDIA_VIDEO" />

    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"
                     android:maxSdkVersion="32" />
    <application ...>
        ...
    </application>
</manifest>

代码部分可以做个判断,例如:

String requestPermission;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
    requestPermission = Manifest.permission.READ_MEDIA_IMAGES;
} else {
    requestPermission = Manifest.permission.READ_EXTERNAL_STORAGE;
}

另外文档建议,如果你的应用只需要访问图片、照片和视频,可以考虑使用照片选择器,而不是声明 READ_MEDIA_IMAGESREAD_MEDIA_VIDEO 权限。

3.访问附近Wi-Fi设备受限

面向Android 13 (API级别33)或更高版本并管理Wi-Fi连接的应用程序必须请求NEARBY_WIFI_DEVICES 运行时权限。这种权限使应用程序访问附近的Wi-Fi设备变得更加容易;在以前版本的Android上,这些应用程序需要声明ACCESS_FINE_LOCATION权限。

需要NEARBY_WIFI_DEVICES权限的API方法:

  • WifiManager:startLocalOnlyHotspot()
  • WifiAwareManager:attach()
  • WifiAwareSession:publish()subscribe()
  • WifiP2pManager:addLocalService()connect()createGroup()discoverPeers()discoverServices()requestDeviceInfo()requestGroupInfo()requestPeers()
  • WifiRttManager:startRanging()

适配说明:

<manifest ...>
    <uses-permission android:name="android.permission.NEARBY_WIFI_DEVICES"
                     android:usesPermissionFlags="neverForLocation" />

    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"
                     android:maxSdkVersion="32" />
</manifest>
  • 如果你的应用不会通过Wi-Fi API推导物理位置,请将usesPermissionFlags属性设为neverForLocation。同时将ACCESS_FINE_LOCATION权限的最高SDK版本设置为32
  • 如果你使用了WifiManagergetScanResults()startScan(),在Android 13还是需要ACCESS_FINE_LOCATION权限,所以去除android:maxSdkVersion="32"

4.广告 ID

使用 Google Play 服务广告 ID 且以 Android 13(API 级别 33)及更高版本为目标平台的应用必须在其清单文件中声明常规AD_ID权限,如下所示:

<manifest ...>
    <uses-permission android:name="com.google.android.gms.permission.AD_ID"/>

    <application ...>
        ...
    </application>
</manifest>

如果您的应用以 Android 13 或更高版本为目标平台且未声明此权限,系统会自动移除广告 ID 并将其替换为一串零。

5.废弃方法

  • PackageManager中的getPackageInfogetApplicationInforesolveActivity等方法。

以常用的getPackageInfo方法举例:

 public static PackageInfo getPackageInfoCompat(PackageManager packageManager, String packageName, int flag)
        throws PackageManager.NameNotFoundException {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
        return packageManager.getPackageInfo(packageName, PackageManager.PackageInfoFlags.of(flag));
    } else {
        return packageManager.getPackageInfo(packageName, flag);
    }
}
  • Intent中的getSerializableExtragetParcelableExtraBundle中的getSerializablegetParcelable等方法。

getParcelableExtra举例说明:

public static <T extends Parcelable> T getParcelableExtraCompat(Intent intent, String name, Class<T> clazz) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
        return intent.getParcelableExtra(name, clazz);
    } else {
        return intent.getParcelableExtra(name);
    }
}

如果你的代码是Kotlin写的,当然也可以将此方法用Kotlin的拓展方法实现,便于替换。

  • WebSettings中的setAppCacheEnabledsetAppCacheMaxSizesetAppCachePath方法被移除。

    setAppCacheEnabled(false) 可以用setCacheMode(WebSettings.LOAD_NO_CACHE)替代。

    setAppCacheEnabled(true) 可以用setCacheMode(WebSettings.LOAD_DEFAULT)替代。

  • 对于以 Android 13(API 级别 33)或更高版本为目标平台的应用,WebSettings中的setForceDark方法已废弃,调用该方法无效。

    WebView 现在始终会根据应用的主题属性 isLightTheme 来设置媒体查询 prefers-color-scheme。换句话说,如果 isLightTheme 为 true 或未指定,则 prefers-color-scheme 为 light;否则为 dark。此行为意味着,系统会自动应用 Web 内容的浅色或深色样式(如果相应内容支持应用主题)。

    对于大多数应用,新行为应自动应用适当的应用样式,不过,您应测试应用以检查是否存在可能已手动控制深色模式设置的情况。

    如果您仍需要自定义应用的颜色主题行为,请改用 setAlgorithmicDarkeningAllowed() 方法。为了向后兼容以前的 Android 版本,我们建议使用 AndroidX 中的等效 setAlgorithmicDarkeningAllowed() 方法。

其他新功能及API

1.更安全地导出上下文注册的接收器

Android 13 允许您指定您应用中的特定广播接收器是否应被导出以及是否对设备上的其他应用可见。如果导出广播接收器,其他应用将可以向您的应用发送不受保护的广播。此导出配置在以 Android 13 或更高版本为目标平台的应用中可用,有助于防止一个主要的应用漏洞来源。在以前的 Android 版本中,设备上的任何应用都可以向动态注册的接收器发送不受保护的广播,除非该接收器受签名权限的保护。

以Android 13或更高版本为目标平台的应用,必须为每个广播接收器指定 RECEIVER_EXPORTEDRECEIVER_NOT_EXPORTED。否则,当您尝试注册广播接收器时,系统会抛出SecurityException

Caused by: java.lang.SecurityException: com.xxx.xxx: One of RECEIVER_EXPORTED or RECEIVER_NOT_EXPORTED should be specified when a receiver isn't being registered exclusively for system broadcasts

当然这个目前不是强制适配的,需要在开发者选项 -> 应用兼容变更中开启DYNAMIC_RECEIVER_EXPLICIT_EXPORT_REQUIRED。这个安全增强措施默认是关闭的,所以暂时无影响。

适配方法:

// 这个广播接收器能够接收来自其他应用程序的广播。
context.registerReceiver(sharedBroadcastReceiver, intentFilter,
                    RECEIVER_EXPORTED);
// 这个私有广播接收器不能够接收来自其他应用的广播。
context.registerReceiver(privateBroadcastReceiver, intentFilter,
                    RECEIVER_NOT_EXPORTED);

因为我们项目之前统一将普通广播替换为了本地广播(LocalBroadcastManager),所以不受影响。

2.预测性返回手势

Android 13(API 级别 33)针对手机、大屏设备和可折叠设备等 Android 设备引入了预测性返回手势。该功能的发布历程跨度将达多年;完全实现后,该功能可让用户在完全完成某个返回手势之前就能预览此手势完成后的目的地或其他结果,以便用户能够决定是继续完成手势还是留在当前视图中。

在这里插入图片描述

参照文档添加对预测性返回手势的支持以及运行官方Codelab发现在Vivo、OPPO、小米上都不起作用,可能被阉割了。。。使用模拟器正常。目前此功能在开发者选项中供测试使用。官方计划在未来的Android版本中向用户提供此界面。

Android 12L开始,发现官方不断地在大屏设备和可折叠设备上提供优化支持。可以看到未来大屏的适配也是一个趋势。

3.各应用语言偏好设定

在许多情况下,多语言用户会将其系统语言设置为某一种语言(例如英语),但又想为特定应用选择其他语言(例如荷兰语、中文或印地语)。为了帮助应用为这些用户提供更好的体验,Android 13 针对支持多种语言的应用引入了以下功能:

  • 系统设置:用户可以在这个集中位置为每个应用选择首选语言。

    您的应用必须在应用的清单中声明 android:localeConfig 属性,以告知系统它支持多种语言。

  • 其他 API:借助这些公共 API(例如 LocaleManager 中的 setApplicationLocales()getApplicationLocales() 方法),应用可以在运行时设置不同于系统语言的其他语言。

    这些 API 会自动与系统设置同步;因此,使用这些 API 创建自定义应用内语言选择器的应用将确保用户获得一致的用户体验,无论他们在何处选择语言偏好设置。公共 API 还有助于减少样板代码量、支持拆分 APK,并且支持应用自动备份,以存储应用级的用户语言设置。

同样,试了几部国产手机此功能都被阉割,所以没有过多的尝试。详情参见文档:各应用语言偏好设定

4.更快断字

断字让分行的文本更易于阅读,并且有助于使界面更具自适应性。从Android 13开始,断字性能提升了高达200%,因此您可以在 TextView 中启用更快断字功能,而几乎不会影响渲染性能。如需启用更快断字功能,请在setHyphenationFrequency() 中使用 fullFastnormalFast 频率。

使用方法:

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
    // 适用于聊天消息
    textView.setHyphenationFrequency(Layout.HYPHENATION_FREQUENCY_NORMAL_FAST);
    // 标准连词符
    textView.setHyphenationFrequency(Layout.HYPHENATION_FREQUENCY_FULL_FAST)
}

下图是使用fullFast属性前后对比:

在这里插入图片描述


在这里插入图片描述

5.带主题的应用图标

从 Android 13 开始,您可以选择启用带主题的应用图标。借助此功能,用户可以调节受支持的Android应用图标色调,以继承所选壁纸和其他主题的配色。

如需支持此功能,您的应用必须提供自适应图标和单色应用图标,并通过清单中的 <adaptive-icon> 元素指向该单色应用图标。如果用户启用了带主题的应用图标(换句话说,在系统设置中开启了带主题的图标切换开关),而启动器支持此功能,则系统将使用用户选择的壁纸和主题来确定色调颜色,然后该颜色将应用于单色应用图标。

在这里插入图片描述

适配方法很简单,只需要额外添加<monochrome/>单色应用图标配置就可以支持这个功能。

<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
    <background android:drawable="@mipmap/ic_launcher_background"/>
    <foreground android:drawable="@mipmap/ic_launcher_foreground"/>
    <monochrome android:drawable="@mipmap/ic_launcher_foreground"/>
</adaptive-icon>

测试了几个部手机都不支持此功能,所以只能使用模拟器测试效果。模拟器效果图:

在这里插入图片描述


有关Android 13的适配内容主要就这么多,更多的变更还是需要我们去阅读官方文档。链接我也贴到了文末。

最后,如果本文对你有所帮助,欢迎收藏点赞。感谢!

参考

  • 官方Android13文档

  • GestureDetector.OnGestureListener overridden methods are not working in Android API 33

  • OPPO - Android 13 应用兼容性适配指导

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

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

相关文章

ChatGPT如何生成可视化图表-示例中国近几年出生人口

本教程收集于&#xff1a;AIGC从入门到精通教程汇总 ChatGPT本身不能直接生成可视化图表&#xff0c;但可以配合其他可视化工具或库 方法一&#xff1a;前端可视化开发库 Echarts&#xff08;地址&#xff1a;Apache ECharts &#xff09; 方法二&#xff1a;现有Python库。…

dump_stack分析函数调用关系实例及其实现

dump_stack分析函数调用关系 文章目录 dump_stack分析函数调用关系一、dump_stack实例二、dump_stack实现分析Step 1: dump_stack_print_infoStep 2: show_stack 三、关于堆栈 一、dump_stack实例 在正点原子阿尔法开发板中查看insmod命令使用什么方法&#xff1a; #include …

[蓝帽杯 2022 初赛]之Misc篇(NSSCTF)刷题记录(复现)⑨

NSSCTF-Misc篇-[蓝帽杯 2022 初赛] 计算机取证&#xff1a;[蓝帽杯 2022 初赛]计算机取证_1[蓝帽杯 2022 初赛]计算机取证_2[蓝帽杯 2022 初赛]计算机取证_3[蓝帽杯 2022 初赛]计算机取证_4 手机取证&#xff1a;[蓝帽杯 2022 初赛]手机取证_1[蓝帽杯 2022 初赛]手机取证_2 网站…

使用堆视图创建3个按钮三角摆放

使用堆视图创建3个按钮三角摆放 效果如图&#xff1a; 分析 从效果图看&#xff0c;想要让3个Button呈三角摆放&#xff0c;需要两个堆视图完成。 首先外部一个大的“垂直堆”&#xff0c;垂直堆第一项放一个Button&#xff0c;第二项放一个“水平堆” 水平堆里再放两个But…

刚入职领导就要求做自动化测试?我懵了,从业务到框架设计总结...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 Python自动化测试&…

深度学习的环境搭建(window+pytorch)

1.检查是否安装CUDA CUDA&#xff08;Compute Unified Device Architecture&#xff09;是由 NVIDIA 推出的一种并行计算平台和编程模型&#xff0c;用于利用 NVIDIA GPU&#xff08;Graphics Processing Unit&#xff09;的强大计算能力进行高性能计算任务。CUDA 的主要特点是…

logstash介绍和使用-ELK文章2

官方 Logstash 是免费且开放的服务器端数据处理管道&#xff0c;能够从多个来源采集数据&#xff0c;转换数据&#xff0c;然后将数据发送到您最喜欢的“存储库”中。 下载和文档&#xff1a;https://www.elastic.co/cn/logstash/ docker部署&#xff1a;https://hub.docker.…

教你快速把heic格式转化jpg,4种方法操作简单

教你快速把heic格式转化jpg的方法&#xff0c;因为HEIC格式图片通常出现在苹果公司的iOS 11操作系统及之后的版本中&#xff0c;这是因为苹果公司在这些版本中采用了HEIF&#xff08;高效图像格式&#xff09;作为默认的照片格式来替代JPEG格式。同时&#xff0c;需要注意的是&…

asp.net+sqlserver社区小区流动人口管理系统

该系统的基本功能包括用户登录&#xff0c;管理员信息管理&#xff0c;社区组织管理&#xff0c;常住人口管理&#xff0c;流动人口管理&#xff0c;社区事务管理&#xff0c;社区服务管理&#xff0c;系统用户管理&#xff0c;修改密码等功能。 &#xff08;3&#xff09;功能…

项目分析v2

用户&#xff1a; 登录&#xff1a; 不能重复登录。 在服务端使用一个hashset记录用户的登录状态&#xff0c;如果用户id不在集合里面&#xff0c;就可以登录&#xff0c;登录时将用户id添加到集合中。用户下线时&#xff0c;将set中的元素删除。 登录成功后&#xff0c;服务端…

【语义分割】LinkNet从0到1和代码实现

文章目录 前言1.网络结构1.1 网络结构示意图1.2 创建LinkNet模型 2.代码2.1 各模块搭建2.1.1 卷积模块2.1.2 反卷积模块2.1.3 编码器模块 2.2 编码网络结构2.3 损失函数&训练2.4 训练 前言 已经有了U-net了&#xff0c;为什么需要linkNet&#xff1f; unet见这个文章【语义…

Docker基础篇(很详细)

一、简单介绍 &#xff08;一&#xff09;为什么用docker 开发人员发开完成就发布一个jar或者war包&#xff0c;其他的都交给运维人员来做&#xff1b;而现在&#xff0c;开发即运维&#xff0c;打包部署上线一套流程走完&#xff1a;开发人员会将项目及其附带的环境一起打包j…

UML类图使用介绍

文章目录 一、UML图1、什么是UML图2、类图概述3、类图的作用 二、类的表示方式举个栗子 三、类与类之间关系的表示方式1、关联关系&#xff08;1&#xff09;单向关联&#xff08;2&#xff09;双向关联&#xff08;3&#xff09;自关联 2、聚合关系3、组合关系4、依赖关系5、继…

安全加密基础—基本概念、keytool、openssl

前言 &#xff08;1&#xff09;本文不涉及源码、底层。只是讲解大概的密码演变过程和基本概念。能让接触到相关名词的人知道这些名词是干嘛的&#xff0c;为什么要有它。专业人士可以当作概念梳理&#xff0c;非专业人士可以当作科普。 &#xff08;2&#xff09;本文你将了解…

亿发工业互联网智能制造ERP系统,生产工厂信息化建设解决方案

亿发工业互联网智能制造ERP系统&#xff0c;生产工厂信息化建设解决方案 随着制造水平的发展&#xff0c;传统工厂原有的生产组织模式和质量管理模式已不能满足先进制造水平的要求。确保公司战略目标的实现&#xff0c;有必要借助信息技术加强对各种业务流程的管理。而企业走向…

Leetcode506. 相对名次

Every day a leetcode 题目来源&#xff1a;506. 相对名次 解法1&#xff1a;STL vector 自定义排序 使用 vector<pair<int, int>> 数组v记录原来 score 数组的下标和成绩。 对v按成绩进行降序排序。 新建一个 vector<string> 数组 ans&#xff0c;遍历…

WiFi(Wireless Fidelity)基础(三)

目录 一、基本介绍&#xff08;Introduction&#xff09; 二、进化发展&#xff08;Evolution&#xff09; 三、PHY帧&#xff08;&#xff08;PHY Frame &#xff09; 四、MAC帧&#xff08;MAC Frame &#xff09; 五、协议&#xff08;Protocol&#xff09; 六、安全&#x…

FP斗篷,2023独立站必看指南

Cloak斗篷技术是一种网络隐身术&#xff0c;通过技术手段实现在网络上匿名和隐身&#xff0c;保护个人隐私和安全。斗篷技术的实现原理是通过使用虚拟专用网络&#xff08;VPN&#xff09;或代理服务器等技术&#xff0c;将用户的真实IP地址隐藏起来&#xff0c;使其在网络上的…

CentOS7离线升级SSH至9.1P1

1、离线安装telnet&#xff08;用telnet登录升级ssh&#xff0c;因为ssh升级时&#xff0c;ssh会话会断开&#xff09; &#xff08;1&#xff09;下载telnet、telnet-server、xinetd&#xff0c;并传到服务器上 http://rpmfind.net/linux/centos/7.9.2009/updates/x86_64/Pack…

目前电视盒子哪个最好?测评20款后整理网络电视盒子推荐

电视盒子是日常必备的数码单品&#xff0c;功能越来越丰富&#xff0c;但很多人不懂要如何选择电视盒子&#xff0c;芯片、内存外还应该考虑系统、技术优化和界面设计等因素。昨天刚刚结束了二十款电视盒子的测评&#xff0c;马上来介绍下我的测评结果&#xff0c;整理了网络电…