Android 支持库迁移到AndroidX

2024-06-17 09:44
文章标签 android androidx 迁移 支持

本文主要是介绍Android 支持库迁移到AndroidX,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

对应官方文档:
developer.android.com/jetpack/and…

简单点说就是,对App开发者而言,AndroidX更加友好,因为我们引入时,只需要关注AndroidX中具体的需要引入的构件版本即可。且大部分具体的构件,具有一致的版本号。开发者使用起来不再需要关注项目自身的最低支持版本和编译版本了,只需要像引入其他的第三方库一样,v1.0、v2.0、v3.0这种方式引入即可。

如原有的引入写法
com.android.support:recyclerview-v7:28.0.0
变成了
androidx.recyclerview:recyclerview:1.0.0

官方文档也提供了Androidx版本具体的升级日志记录。
developer.android.com/jetpack/and… developer.android.com/jetpack/and…

三、支持库迁移到AndroidX

3.1 迁移AndroidX的必要性

AndroidX对开发者使用更加友好,同时,支持库文档上官方已经明确支持库后续不再维护。另外,在Android Studio上新建模块时,也发现如果没有迁移到AndroidX,模块创建不了,表明开始有强制性的措施使得开发者必须迁移到AndroidX。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

3.2 迁移AndroidX的前置条件

Android Sudio在3.2版本开始,对直接迁移到AndroidX进行了支持。在操作路径Refactor > Migrate to AndroidX下,但使用时会发现可能存在如下提示:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

这也说明了,利用官方内置的迁移方式迁移AndroidX之前,工程环境上最好满足如下条件:
1,Android Studio 3.2及以上。当前时点最新版本已经是3.5稳定版了。
2,AGP版本3.2.0及以上,对应的Gradle版本4.6及以上。
3,项目编译版本28及以上。

如果当前项目没有满足上述条件,可以先升级对应的配套。

3.3 迁移过程

Android官方提供了具体的迁移指引。具体参见文档:
developer.android.com/jetpack/and…

Just Start!

以下主要记录实际项目中的迁移过程,以及遇到的问题及解决。

Refactor > Migrate to AndroidX操作后,AS会有对应的迁移提醒,提示你去备份项目文件,如有必要可以先备份。但一般而言,AS项目都是基于Git进行管理,直接单独切一个分支进行迁移操作即可,此处备份成zip现实意义不大。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

点击Migrate后,会出现弹窗Looking for Usages,开始在当前项目中搜索所有可能需要迁移的源文件,包括代码源文件、XML文件、build.gradle配置文件等,最终会列出当前主工程使用到支持库的所有文件列表。

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

点击Do Refactor确认迁移,AS自动执行迁移AndroidX的替换过程。如将对应的支持库类名、包名、构件名等都替换成相应的AndroidX形式。

一点时间后,主工程替换完成。此时打开gradle.properties,会发现自动添加了如下配置项。

android.useAndroidX=true
android.enableJetifier=true

android.useAndroidX=true,表示主工程使用AndroidX形式。
android.enableJetifier=true,表示针对主工程中使用到的三方库,也会自动执行AndroidX的替换过程。

同时,在自动执行三方库的替换时,出下了如下报错信息:

ERROR: Unable to resolve dependency for ‘:MyCorn@prodDebug/compileClasspath’: Failed to transform file ‘fingerprint-1.1.1.aar’ to match attributes {artifactType=processed-aar} using transform JetifyTransform

大致的意思是使用JetifyTransformfingerprint-1.1.1.aar进行替换过程中,出现了问题。但具体问题没有进一步的提示信息。于是,直接通过命令执行下构建看一下:

./gradlew assembleDevDebug

1: Task failed with an exception.

  • What went wrong:
    Could not resolve all files for configuration ‘:MyCorn:devDebugCompileClasspath’.

Failed to transform file ‘fingerprint-1.1.1.aar’ to match attributes {artifactType=processed-aar} using transform JetifyTransform
Failed to transform ‘/Users/corn/.gradle/caches/modules-2/files-2.1/com.corn.feature/fingerprint/1.1.1/ae2da4c824fb2923eac7a1340222d50d6308f7ea/fingerprint-1.1.1.aar’ using Jetifier. Reason: 8. (Run with --stacktrace for more details.) To disable Jetifier, set android.enableJetifier=false in your gradle.properties file.

进而,带上–stacktrace看看。

./gradlew assembleDevDebug --stacktrace


Caused by: java.lang.ArrayIndexOutOfBoundsException: 8
at org.objectweb.asm.ClassReader.readFrameType(ClassReader.java:2313)
at org.objectweb.asm.ClassReader.readFrame(ClassReader.java:2269)
at org.objectweb.asm.ClassReader.readCode(ClassReader.java:1448)
at org.objectweb.asm.ClassReader.readMethod(ClassReader.java:1126)
at org.objectweb.asm.ClassReader.accept(ClassReader.java:698)
at org.objectweb.asm.ClassReader.accept(ClassReader.java:500)
at com.android.tools.build.jetifier.processor.transform.bytecode.ByteCodeTransformer.runTransform(ByteCodeTransformer.kt:39)
at com.android.tools.build.jetifier.processor.Processor.visit(Processor.kt:328)
at com.android.tools.build.jetifier.processor.archive.ArchiveFile.accept(ArchiveFile.kt:41)
at com.android.tools.build.jetifier.processor.Processor.visit(Processor.kt:316)
at com.android.tools.build.jetifier.processor.archive.Archive.accept(Archive.kt:66)
at com.android.tools.build.jetifier.processor.Processor.visit(Processor.kt:316)
at com.android.tools.build.jetifier.processor.archive.Archive.accept(Archive.kt:66)
at com.android.tools.build.jetifier.processor.Processor.transformLibrary(Processor.kt:312)
at com.android.tools.build.jetifier.processor.Processor.transform(Processor.kt:175)
at com.android.build.gradle.internal.dependency.JetifyTransform.transform(JetifyTransform.kt:199)
… 39 more

我们发现,JetifyTransform内部使用了ASM,在对aar进行ClassReader的过程中抛出了异常。并且从错误栈信息上看,应该有一类叫jetifier的工具,是在这个工具中调用的ASM操作。

官方文档搜索下,果然发现了jetifier的踪迹。
developer.android.com/studio/comm…

同样的,Google Source上也找到了其对应的实现。
android.googlesource.com/platform/fr…

下载对应的jetifier-standalone,解压后,执行命令对fingerprint-1.1.1.aar执行AndroidX转化。

➜ bin ./jetifier-standalone -i ./fingerprint-1.1.1.aar -o 11.aar
Exception in thread “main” java.lang.ArrayIndexOutOfBoundsException: 8
at org.objectweb.asm.ClassReader.readFrameType(ClassReader.java:2313)
at org.objectweb.asm.ClassReader.readFrame(ClassReader.java:2269)
at org.objectweb.asm.ClassReader.readCode(ClassReader.java:1448)
at org.objectweb.asm.ClassReader.readMethod(ClassReader.java:1126)
at org.objectweb.asm.ClassReader.accept(ClassReader.java:698)
at org.objectweb.asm.ClassReader.accept(ClassReader.java:500)
at com.android.tools.build.jetifier.processor.transform.bytecode.ByteCodeTransformer.runTransform(ByteCodeTransformer.kt:40)
at com.android.tools.build.jetifier.processor.Processor.visit(Processor.kt:539)
at com.android.tools.build.jetifier.processor.archive.ArchiveFile.accept(ArchiveFile.kt:53)
at com.android.tools.build.jetifier.processor.Processor.visit(Processor.kt:521)
at com.android.tools.build.jetifier.processor.archive.Archive.accept(Archive.kt:76)
at com.android.tools.build.jetifier.processor.Processor.visit(Processor.kt:521)
at com.android.tools.build.jetifier.processor.archive.Archive.accept(Archive.kt:76)
at com.android.tools.build.jetifier.processor.Processor.transformLibrary(Processor.kt:517)
at com.android.tools.build.jetifier.processor.Processor.transform2(Processor.kt:291)
at com.android.tools.build.jetifier.processor.Processor.transform2 d e f a u l t ( P r o c e s s o r . k t : 251 ) a t c o m . a n d r o i d . t o o l s . b u i l d . j e t i f i e r . s t a n d a l o n e . M a i n . r u n ( M a i n . k t : 156 ) a t c o m . a n d r o i d . t o o l s . b u i l d . j e t i f i e r . s t a n d a l o n e . M a i n default(Processor.kt:251) at com.android.tools.build.jetifier.standalone.Main.run(Main.kt:156) at com.android.tools.build.jetifier.standalone.Main default(Processor.kt:251)atcom.android.tools.build.jetifier.standalone.Main.run(Main.kt:156)atcom.android.tools.build.jetifier.standalone.MainCompanion.main(Main.kt:109)
at com.android.tools.build.jetifier.standalone.Main.main(Main.kt)

发现出现了同样的错误信息。

显然,应该是fingerprint-1.1.1.aar中有字节码有问题。经查,fingerprint内部直接以jar方式引入了三星的指纹识别库,已经很比较老的版本了,经业务同学确认,现在已经可以直接去除。

去除fingerprint内部的三星指纹库后,升级版本,下载对应的aar文件后,再次尝试转化:

./jetifier-standalone -i ./fingerprint-1.1.3-20190916.092208-1.aar -o mm.aar

执行成功,且有转换后的对应文件生成。

主工程更新fingerprint对应依赖版本后,重新执行构建,出现错误提示:

e: /Users/corn/AndroidStudioProjects/MyCorn/base/src/main/java/com/mycorn/base/mvvm/EventLiveData.kt: (13, 5): ‘observe’ overrides nothing
e: /Users/corn/AndroidStudioProjects/MyCorn/base/src/main/java/com/mycorn/base/mvvm/EventLiveData.kt: (20, 5): ‘removeObserver’ overrides nothing

原因在于对应的LiveData接口observe、removeObserver中的形参有所改动,从原来的

@NonNull Observer observer

变成了

@NonNull Observer<? super T> observer

修正EventLiveData类中的重写方法的对应形参,与接口保持一致即可。

再次重新构建,出现错误信息:

/Users/corn/AndroidStudioProjects/MyCorn/trans/src/main/java/com/mycorn/biz/supertrans/v12/slide/ItemSlideHelper.java:566: 错误: 程序包androidx.appcompat.recyclerview.R不存在
.getDimension(androidx.appcompat.recyclerview.R.dimen.item_touch_helper_swipe_escape_velocity);

/Users/corn/AndroidStudioProjects/MyCorn/trans/src/main/java/com/mycorn/biz/supertrans/v12/slide/ItemSlideHelper.java:568: 错误: 程序包androidx.appcompat.recyclerview.R不存在
.getDimension(androidx.appcompat.recyclerview.R.dimen.item_touch_helper_swipe_escape_max_velocity);

/Users/corn/AndroidStudioProjects/MyCorn/trans/src/main/java/com/mycorn/biz/supertrans/v12/slide/ItemSlideHelper.java:2115: 错误: 程序包androidx.appcompat.recyclerview.R不存在
androidx.appcompat.recyclerview.R.dimen.item_touch_helper_max_drag_scroll_per_frame);

核查官方文档,对应的替换关系应该是:

ndroid.support.v7.recyclerview.R

替换为

androidx.recyclerview.R

此处替换成了

androidx.appcompat.recyclerview.R

按照文档对应修正过来。

再次重新构建,可以构建成功。

3.4 核验与发现

此时构建成功,是不是就真的都处理好了呢,是不是都没问题了呢。显然不是的。

首先,分别查看主工程对应的编译时和运行时依赖,看看是否都从android.support.* 替换成了androidx.*

./gradlew -q MyCorn:dependencies --configuration devDebugAndroidTestCompileClasspath > ~/compile.txt

./gradlew -q MyCorn:dependencies --configuration devDebugAndroidTestRuntimeClasspath > ~/runtime.txt

仔细对比后发现,虽然依赖中都已经被替换成了androidx.* 。但编译时的依赖中含有大量的rc01。如:

androi
dx.appcompat:appcompat:1.0.0-rc01
androidx.recyclerview:recyclerview:1.0.0-rc01

但运行时依赖中却没有rc01

androidx.appcompat:appcompat:1.0.0
androidx.recyclerview:recyclerview:1.0.0

经核查后发现,主工程中之前依赖支持库时有两种写法,一种是直接写法,如:

implementation 'com.android.support:appcompat-v7:28.0.0

另一种是采取统一定义后,进行的变量形式引入:

api rootProject.ext.dependencies[“appcompat-v7”]

其中,具体变量,如appcompat-v7被统一定义在了专用的一个dependencies.gradle文件中。
大致形式如下:

// 统一配置管理
def supportVersion = “28.0.0”

project.ext {
android = [
“compileSdkVersion”: 28,
“minSdkVersion” : 19,
“targetSdkVersion” : 26,
“javaMaxHeapSize” : “5G”
]

dependencies = [
“support-compat” : “com.android.support:support-compat: s u p p o r t V e r s i o n " , " s u p p o r t − c o r e − u t i l s " : " c o m . a n d r o i d . s u p p o r t : s u p p o r t − c o r e − u t i l s : {supportVersion}", "support-core-utils" : "com.android.support:support-core-utils: supportVersion","supportcoreutils":"com.android.support:supportcoreutils:{supportVersion}”,
“support-core-ui” : “com.android.support:support-core-ui: s u p p o r t V e r s i o n " , " s u p p o r t − m e d i a − c o m p a t " : " c o m . a n d r o i d . s u p p o r t : s u p p o r t − m e d i a − c o m p a t : {supportVersion}", "support-media-compat": "com.android.support:support-media-compat: supportVersion","supportmediacompat":"com.android.support:supportmediacompat:{supportVersion}”,
“support-fragment” : “com.android.support:support-fragment: s u p p o r t V e r s i o n " , " s u p p o r t − a n n o t a t i o n s " : " c o m . a n d r o i d . s u p p o r t : s u p p o r t − a n n o t a t i o n s : {supportVersion}", "support-annotations" : "com.android.support:support-annotations: supportVersion","supportannotations":"com.android.support:supportannotations:{supportVersion}”,
“appcompat-v7” : “com.android.support:appcompat-v7:${supportVersion}”,


]
}

来到对应的文件,发现对应的android.support.*并没有被替换成androidx.*

而在build.gralde中直接引入的写法,是被相应替换了的。已经被正确替换成了:

implementation ‘androidx.appcompat:appcompat:1.0.0’

也就是说,构建时,遇到api rootProject.ext.dependencies["appcompat-v7"],其实是没有被准确识别的,按照编译时的依赖关系,最终被识别成了带有rc01的AndroidX版本。

于是,才出现了编译时和运行时大量的版本不一致情况。

解决起来很简单,直接在dependencies.gradle文件中,将android.support.*对应人为替换成androidx.*形式,并重新规范命名和版本,相应修正对应的被使用到的地方,并修正成统一的AndroidX依赖写法。

// 统一配置管理
def androidXVersion = “1.0.0”

project.ext {
android = [
“compileSdkVersion”: 28,
“minSdkVersion” : 19,
“targetSdkVersion” : 26,
“javaMaxHeapSize” : “5G”
]

dependencies = [
“androidx-core” : “androidx.core:core: a n d r o i d X V e r s i o n " , " a n d r o i d x − c o r e − u t i l s " : " a n d r o i d x . l e g a c y : l e g a c y − s u p p o r t − c o r e − u t i l s : {androidXVersion}", "androidx-core-utils" : "androidx.legacy:legacy-support-core-utils: androidXVersion","androidxcoreutils":"androidx.legacy:legacysupportcoreutils:{androidXVersion}”,
“androidx-core-ui” : “androidx.legacy:legacy-support-core-ui: a n d r o i d X V e r s i o n " , " a n d r o i d x − m e d i a " : " a n d r o i d x . m e d i a : m e d i a : {androidXVersion}", "androidx-media" : "androidx.media:media: androidXVersion","androidxmedia":"androidx.media:media:{androidXVersion}”,
“androidx-fragment” : “androidx.fragment:fragment: a n d r o i d X V e r s i o n " , " s u p p o r t − a n n o t a t i o n s " : " a n d r o i d x . a n n o t a t i o n : a n n o t a t i o n : {androidXVersion}", "support-annotations" : "androidx.annotation:annotation: androidXVersion","supportannotations":"androidx.annotation:annotation:{androidXVersion}”,
“androidx-appcompat” : “androidx.appcompat:appcompat:${androidXVersion}”,


]
}

重新输出编译时和运行时依赖,发现此时支持库版本已经一致。

重新构建项目,发现可以构建成功。

但到此时,我们依然有四个问题需要进一步确认:
1,主工程中是否有支持库相关的一些特别的写法,结果会跟上面的dependencies.gradle一样,不能被自动识别并迁移?例如反射?字符串?甚至字符串拼接?等等。

2,原有依赖库android.support.*肯定会有一些混淆配置,现在迁移成androidx.*后,混淆配置这块如何对应处理?

droidx-appcompat" : “androidx.appcompat:appcompat:${androidXVersion}”,


]
}

重新输出编译时和运行时依赖,发现此时支持库版本已经一致。

重新构建项目,发现可以构建成功。

但到此时,我们依然有四个问题需要进一步确认:
1,主工程中是否有支持库相关的一些特别的写法,结果会跟上面的dependencies.gradle一样,不能被自动识别并迁移?例如反射?字符串?甚至字符串拼接?等等。

2,原有依赖库android.support.*肯定会有一些混淆配置,现在迁移成androidx.*后,混淆配置这块如何对应处理?

这篇关于Android 支持库迁移到AndroidX的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

公共筛选组件(二次封装antd)支持代码提示

如果项目是基于antd组件库为基础搭建,可使用此公共筛选组件 使用到的库 npm i antdnpm i lodash-esnpm i @types/lodash-es -D /components/CommonSearch index.tsx import React from 'react';import { Button, Card, Form } from 'antd'

(超详细)YOLOV7改进-Soft-NMS(支持多种IoU变种选择)

1.在until/general.py文件最后加上下面代码 2.在general.py里面找到这代码,修改这两个地方 3.之后直接运行即可

Eclipse+ADT与Android Studio开发的区别

下文的EA指Eclipse+ADT,AS就是指Android Studio。 就编写界面布局来说AS可以边开发边预览(所见即所得,以及多个屏幕预览),这个优势比较大。AS运行时占的内存比EA的要小。AS创建项目时要创建gradle项目框架,so,创建项目时AS比较慢。android studio基于gradle构建项目,你无法同时集中管理和维护多个项目的源码,而eclipse ADT可以同时打开

android 免费短信验证功能

没有太复杂的使用的话,功能实现比较简单粗暴。 在www.mob.com网站中可以申请使用免费短信验证功能。 步骤: 1.注册登录。 2.选择“短信验证码SDK” 3.下载对应的sdk包,我这是选studio的。 4.从头像那进入后台并创建短信验证应用,获取到key跟secret 5.根据技术文档操作(initSDK方法写在setContentView上面) 6.关键:在有用到的Mo

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

Android我的二维码扫描功能发展史(完整)

最近在研究下二维码扫描功能,跟据从网上查阅的资料到自己勉强已实现扫描功能来一一介绍我的二维码扫描功能实现的发展历程: 首页通过网络搜索发现做android二维码扫描功能看去都是基于google的ZXing项目开发。 2、搜索怎么使用ZXing实现自己的二维码扫描:从网上下载ZXing-2.2.zip以及core-2.2-source.jar文件,分别解压两个文件。然后把.jar解压出来的整个c

android 带与不带logo的二维码生成

该代码基于ZXing项目,这个网上能下载得到。 定义的控件以及属性: public static final int SCAN_CODE = 1;private ImageView iv;private EditText et;private Button qr_btn,add_logo;private Bitmap logo,bitmap,bmp; //logo图标private st

Android多线程下载见解

通过for循环开启N个线程,这是多线程,但每次循环都new一个线程肯定很耗内存的。那可以改用线程池来。 就以我个人对多线程下载的理解是开启一个线程后: 1.通过HttpUrlConnection对象获取要下载文件的总长度 2.通过RandomAccessFile流对象在本地创建一个跟远程文件长度一样大小的空文件。 3.通过文件总长度/线程个数=得到每个线程大概要下载的量(线程块大小)。

时间服务器中,适用于国内的 NTP 服务器地址,可用于时间同步或 Android 加速 GPS 定位

NTP 是什么?   NTP 是网络时间协议(Network Time Protocol),它用来同步网络设备【如计算机、手机】的时间的协议。 NTP 实现什么目的?   目的很简单,就是为了提供准确时间。因为我们的手表、设备等,经常会时间跑着跑着就有误差,或快或慢的少几秒,时间长了甚至误差过分钟。 NTP 服务器列表 最常见、熟知的就是 www.pool.ntp.org/zo

高仿精仿愤怒的小鸟android版游戏源码

这是一款很完美的高仿精仿愤怒的小鸟android版游戏源码,大家可以研究一下吧、 为了报复偷走鸟蛋的肥猪们,鸟儿以自己的身体为武器,仿佛炮弹一样去攻击肥猪们的堡垒。游戏是十分卡通的2D画面,看着愤怒的红色小鸟,奋不顾身的往绿色的肥猪的堡垒砸去,那种奇妙的感觉还真是令人感到很欢乐。而游戏的配乐同样充满了欢乐的感觉,轻松的节奏,欢快的风格。 源码下载