当支付宝|当支付宝 App 遇见 AndroidX......

当支付宝|当支付宝 App 遇见 AndroidX......
文章图片

作者:扬州
本文主要介绍支付宝Android端拥抱AndroidX过程中的一些新(xin)鲜(suan)事(lei),通过文章可以了解到以下内容:

  • 支付宝升级到AndroidX有何不同/难点?
  • Android Studio的Migrate AndroidX是如何实现的?
  • APK产物怎么适配AndroidX?
背景 自2018年Android Support被“革了命”后,Support扩展库被尘封,享年28(最后的一个官方版本)。而AndroidX得到了社区的广泛接纳,众多知名框架纷纷推出AndroidX适配的版本,各大App厂家也跟进AndroidX升级。
不负责任的说,缺失AndroidX长远来看将与Android社区组件脱节,无法享受到组件的维护,新的组件也难以融入,比如Paging 3.0,CameraX 1.0.0,Hilt 1.0.0,AppSearch 1.0.0,更不用说后起新秀Jetpack Compose。
AndroidX 业内情况 这里不细数AndroidX的内容,一句话概括:AndroidX是Support的接班人,承接了原有Support 28的功能,并且持续迭代出了更多的扩展库。
通过查看某应用市场,榜单50的App软件包,分析这些APK的AndroidX适配情况,我们得到了一个数据,目标群体排序前50的有70%适配了AndroidX(列表并不权威,不必过于关注排名):
应用数 适配AndroidX 百分比
50 35 70%
应用宝列表:[分类页面 - 应用首页]()
业内App适配AndroidX情况还是比较好的,虽然抽样有70%适配了,那剩下的也有30%,而且都是体量靠前的App。
读到这儿,你可能就要问了:为什么还有多应用还没有适配?
在软件开发中经常会提到“技术债”,也就是说每次面临系统重大调整,对研发人员都是一种考验。具体到每个独立的App,架构差异、业务研发模式、组件使用诉求、研发饱和情况等,都会形成升级的阻力。
那么,同样会面临以上阻力的支付宝App,遇见AndroidX,会擦出怎样的“火花”?
钱包适配AndroidX 为了让庞大的支付宝App顺利披上AndroidX的战袍,我们需要设计一个技术方案,能够灵活和众多业务进行适配工作,有意愿和诉求的业务,进行深度源码适配;暂时“实力不允许”的业务,我们想办法提供兜底的方案。
AndroidX适配点 适配AndroidX,通过分析方案,我们知道主要处理以下内容:
  1. Java/Class源码,各种support的类,包替换
  2. xml布局,替换support组件名字
  3. build.gradle,修改dependency中的GAV
  4. proguard,混淆规则替换
  5. pom依赖,dependency处理
钱包架构现状 蚂蚁的Android开发对支付宝的构建体系都非常熟悉,他与原生是有一些差异的。这里我们简单做个抽象,以便非蚂蚁的Android开发能快速了解一个全貌。
从项目的构建上来看,我们可以画出两张图,我给取名分别是:“传统Android开发”,“支付宝Android开发”。这里不讨论Android的插件化(这几乎是国内特有的风气)。
当支付宝|当支付宝 App 遇见 AndroidX......
文章图片

PS:这里没有涵盖Google Play 2021.8月即将强推的 AAB上架方案,感兴趣的同学可以自己脑补下大图。
当支付宝|当支付宝 App 遇见 AndroidX......
文章图片

对比两张图,主要区别在用AGP构建的差异,mPaas目前也是钱包体系商业化的一个输出。依托于深度的构建定制,结合钱包的运行时框架,我们充分的实现了业务的并行开发和和动态交付能力。如果要用几个简单的词汇来描述,我会这样表述:
  • 业务隔离 =》Bundle
  • 构建分离 =》mPaas+Bundle
  • 产物聚合,动态发布 =》mPaas+Portal+Bundle
这个体系优点很多,前人种树后人乘凉,但是也需要承担深度定制后遗留的痛点和坑位。
了解完了AndroidX的适配点和钱包的开发模式,你因该能猜到,钱包要适配AndroidX的遇到的第一个问题:适配需要Java源码或者Class字节码的修改,而钱包中业务是通过APK隔离的,几乎没有AAR。
阻力与动力 在支付宝这边,对于Android开发来说,我们是有对标原生开发的,包括研发效能,编译链对齐等。
我们升级AndroidX的阻力包括:
  • Support类库广泛使用,API兼容性挑战;
  • 众多跨团队模块,改一下就是“百团大战”;
  • 官方升级机制不适用,不适用钱包的构建;
  • 布局动态容器,存在冲突。
动力也不少(不分先后):
  • Android Support已不再维护;
  • AndroidX社区融入度提升;
  • 现代Android开发基于AndroidX;
  • 业务上有使用的诉求。
【当支付宝|当支付宝 App 遇见 AndroidX......】在多次的沟通和调整后,我们的研究的方案几个点要保证:
  • 关联方尽量少;
  • 兼容能力;
  • 侵入性小;
  • 可插拔的适配。
不能要求所有业务开发一起源码适配AndroidX,否则你可能会问自己:我是谁?我在哪?我要干什么?
需要兼容本身就很少迭代的历史模块,比如16年开始的就没有变更的模块,你要找人配合是不是现实?
当支付宝|当支付宝 App 遇见 AndroidX......
文章图片

站在业务开发的立场,适配不要比官方方案更复杂,尽量做到无感知适配。适配能力向Andoird靠拢,提供开关,方便业务选择性处理。
钱包适配实现 这里是一张早先绘制的,在构建中适配AndroidX的流程和卡点。
当支付宝|当支付宝 App 遇见 AndroidX......
文章图片

类替换
Java/Class代码处理是针对与源码工程和AAR的,钱包的bundle是apk,内部是编译后的dex和二进制xml。在类替换环节,我们通过解析dex文件结构,将support替换集合中的映射关系逐一修改到dex中。
xml替换
apk中的xml是二进制的的plain格式,他不是文本文件,有自己的格式。这个格式简单来说就是Chunk,我们找到其中的存放的xml node节点,把节点对应的值运用support替换集合,修改完后重新保存为二进制的xml。
Bundle替换
结合前面两个步骤,得到一个新的apk。替换dependency中的bundle/devbundle依赖。
产物
当我们对一个业开启适配后,他的产物会发生变化。
当支付宝|当支付宝 App 遇见 AndroidX......
文章图片

当支付宝|当支付宝 App 遇见 AndroidX......
文章图片

修改完成后,我们会输出一份txt报告,共开发者检查使用,这里有一份示例:
OLD: /Users/█████/work/android-phone-█████-git/build/build/outputs/apk/debug/█████-build-debug.apk NEW: /Users/█████/work/android-phone-█████-git/build/build/bundle-cache/█████-build-debug.apkclass count: 53 xml count: 1================= ====DEX==== =================* com/alipay/android/█████/█████/█████ (3) * com/alipay/android/█████/██████████$█████ (3) * com/alipay/mobile/██████████/███/██████████$12 (9) * com/alipay/mobile/█████/█/█████ (45) * com/alipay/mobile/██/██/█████$██████████ (28) * com/alipay/mobile/█/███/███████████████ (3) * com/alipay/android/█████/█████/███████ (3) * com/alipay/mobile/██████/█/██████$████ (9) * com/alipay/mobile/███/███/█████████ (13) * com/alipay/android/███/██/█████ (20) * com/alipay/mobile/███/█/████$█████ (9) * com/alipay/android/███/█████████$█████ (12) * com/alipay/mobile/██/█/██████████████ (27) * com/alipay/mobile/█/███/███████████████ (3) * com/alipay/mobile/██/██/█████/████████ (9) * com/alipay/android/██/█/██████████████████ (6) * com/alipay/mobile/█/███/███████████████ (3) * com/alipay/android/█████/███/██████ (23) * com/alipay/android/██/██/████████████ (30) * com/huawei/android/█████/████████$███████████ (2880) * com/alipay/android/████/██████$14 (4) * com/alipay/android/█████/██/█████████ (41) * com/alipay/android/████/████/██████████████████ (129) * com/alipay/android/████/████████████ (22) * com/alipay/android/████████/███████/█████████████████ (3) * com/alipay/mobile/████/██████/███████/█████████████████████████ (3) * com/alipay/android/████████/████/███████████████ (18) * com/alipay/mobile/████████/███/████████████████████████ (27) * com/alipay/android/████████/██████████████████$21 (17) * com/alipay/mobile/██████/██████/████████████████████████████████$██████████████████████ (6) * com/alipay/android/████████/██████████████████$22 (8) * com/alipay/mobile/████████/███/██████████████████$5 (9) * com/alipay/android/████████/██████████████████$20 (6) * com/alipay/android/████████/████/█████████████ (23) * com/alipay/mobile/██████/██████/██████████████████████████$4 (3) * com/alipay/mobile/██████/██████/████████████████████████████████$████████████████████ (38) * com/alipay/mobile/██████/██████/██████████████████████████$2 (4) * com/alipay/mobile/██████/██████/██████████████████████████$8 (6) * com/alipay/android/████████/██████████████████ (293) * com/alipay/mobile/██████/██████/██████████████████████████$6 (3) * com/alipay/mobile/██████/██████/█████████/███████████████$5 (9) * com/alipay/android/████/████████████$1$1 (11) * com/alipay/mobile/██████/██████/████████████████████████████████ (87) * com/alipay/android/████████/█████/██████████████████████ (7) * com/alipay/android/████████/tab/█████████████████$2 (4) * com/alipay/android/████████/tab/█████████████████$1 (4) * com/alipay/mobile/██████/██████/████████████████████████████████$2$1 (7) * com/alipay/android/████/████████████$1 (5) * com/alipay/android/████████/████/███████████████$████████████████████$1$1 (9) * com/alipay/mobile/████████/███/████████████$5 (9) * com/alipay/mobile/████/████████/█████████████████████ (2) * com/alipay/mobile/████████/███/████████████████████████$4 (9) * com/alipay/mobile/██████/██████/██████████████████████████ (306)Reference detail: * com/alipay/android/████████/█████/█████████████████████ (3) | android/support/annotation/NonNull -> androidx/annotation/NonNull (3)* com/alipay/android/████████/██████████████████$█████████████ (3) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (3)* com/alipay/mobile/████████/███/████████████$12 (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/mobile/████████/███/████████████ (45) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (45)* com/alipay/mobile/██████/██████/██████████████████████████$LoginActivityOnWindowsFocusChangeReceiver (28) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (19)* com/alipay/mobile/████/██████/█████████████████████████ (3) | android/support/annotation/NonNull -> androidx/annotation/NonNull (3)* com/alipay/android/████████/███████/████████████ (3) | android/support/annotation/Nullable -> androidx/annotation/Nullable (3)* com/alipay/mobile/████████/███/████████████$█████████████ (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/mobile/██████/██████/██████████████████████ (13) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (10) | android/support/annotation/NonNull -> androidx/annotation/NonNull (3)* com/alipay/android/████████/████/████████████████████████ (20) | android/support/v4/app/FragmentPagerAdapter -> androidx/fragment/app/FragmentPagerAdapter (3) | android/support/v4/app/FragmentManager -> androidx/fragment/app/FragmentManager (11) | android/support/v4/app/Fragment -> androidx/fragment/app/Fragment (6)* com/alipay/mobile/████████/███/████████████████████████$█████████████ (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/android/████████/██████████████████$TabChangeTimeRunnable (12) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (12)* com/alipay/mobile/████████/███/██████████████████ (27) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (27)* com/alipay/mobile/base/config/ConfigServiceLmacSyncCallback (3) | android/support/annotation/NonNull -> androidx/annotation/NonNull (3)* com/alipay/mobile/██████/██████/█████████/███████████████ (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/android/████████/████/█████████████████████████████████████████ (6) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (6)* com/alipay/mobile/█████/█████/████████████████████ (3) | android/support/annotation/NonNull -> androidx/annotation/NonNull (3)* com/alipay/android/████████/████/███████████████ (23) | android/support/v4/app/Fragment -> androidx/fragment/app/Fragment (7) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (16)* com/alipay/android/████████/████/████████████████████ (30) | android/support/v4/view/ViewPager -> androidx/viewpager/widget/ViewPager (18) | android/support/v4/app/FragmentManager -> androidx/fragment/app/FragmentManager (7) | android/support/v4/view/PagerAdapter -> androidx/viewpager/widget/PagerAdapter (5)* com/huawei/android/██████████/███████████████$████████████████████ (2880) | android/support/v4/view/ViewPager -> androidx/viewpager/widget/ViewPager (277) | android/support/v4/app/FragmentManager -> androidx/fragment/app/FragmentManager (166) | android/support/v4/app/Fragment -> androidx/fragment/app/Fragment (270) | android/support/v4/view/PagerAdapter -> androidx/viewpager/widget/PagerAdapter (92) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (1018) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (542) | android/support/v4/view/ViewPager$OnPageChangeListener -> androidx/viewpager/widget/ViewPager$OnPageChangeListener (105) | android/support/v4/app/FragmentTransaction -> androidx/fragment/app/FragmentTransaction (144) | android/support/v4/util/ArrayMap -> androidx/collection/ArrayMap (106) | android/support/v4/app/FragmentPagerAdapter -> androidx/fragment/app/FragmentPagerAdapter (52) | android/support/v4/content/ContextCompat -> androidx/core/content/ContextCompat (48) | android/support/annotation/Nullable -> androidx/annotation/Nullable (26) | android/support/annotation/NonNull -> androidx/annotation/NonNull (34)* com/alipay/android/████████/██████████████████$14 (4) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (4)* com/alipay/android/████████/tab/█████████████████ (41) | android/support/v4/util/ArrayMap -> androidx/collection/ArrayMap (41)* com/alipay/android/████████/████/██████████████████ (129) | android/support/v4/view/ViewPager -> androidx/viewpager/widget/ViewPager (69) | android/support/v4/view/ViewPager$OnPageChangeListener -> androidx/viewpager/widget/ViewPager$OnPageChangeListener (39) | android/support/v4/view/PagerAdapter -> androidx/viewpager/widget/PagerAdapter (21)* com/alipay/android/████/████████████ (22) | android/support/v4/app/Fragment -> androidx/fragment/app/Fragment (5) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (14) | android/support/annotation/Nullable -> androidx/annotation/Nullable (3)* com/alipay/android/████████/███████/█████████████████ (3) | android/support/annotation/Nullable -> androidx/annotation/Nullable (3)* com/alipay/mobile/████/██████/███████/█████████████████████████ (3) | android/support/annotation/NonNull -> androidx/annotation/NonNull (3)* com/alipay/android/████████/████/███████████████ (18) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (18)* com/alipay/mobile/████████/███/████████████████████████ (27) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (27)* com/alipay/android/████████/██████████████████$21 (17) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (17)* com/alipay/mobile/██████/██████/████████████████████████████████$██████████████████████ (6) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (6)* com/alipay/android/████████/██████████████████$22 (8) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (8)* com/alipay/mobile/████████/███/██████████████████$5 (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/android/████████/██████████████████$20 (6) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (6)* com/alipay/android/████████/████/█████████████ (23) | android/support/v4/view/ViewPager$OnPageChangeListener -> androidx/viewpager/widget/ViewPager$OnPageChangeListener (7) | android/support/v4/view/ViewPager -> androidx/viewpager/widget/ViewPager (16)* com/alipay/mobile/██████/██████/██████████████████████████$4 (3) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (3)* com/alipay/mobile/██████/██████/████████████████████████████████$████████████████████ (38) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (29) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/mobile/██████/██████/██████████████████████████$2 (4) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (4)* com/alipay/mobile/██████/██████/██████████████████████████$8 (6) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (6)* com/alipay/android/████████/██████████████████ (293) | android/support/v4/app/Fragment -> androidx/fragment/app/Fragment (23) | android/support/v4/util/ArrayMap -> androidx/collection/ArrayMap (3) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (184) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (77) | android/support/annotation/Nullable -> androidx/annotation/Nullable (6)* com/alipay/mobile/██████/██████/██████████████████████████$6 (3) | android/support/annotation/Nullable -> androidx/annotation/Nullable (3)* com/alipay/mobile/██████/██████/█████████/███████████████$5 (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/android/████/████████████$1$1 (11) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (11)* com/alipay/mobile/██████/██████/████████████████████████████████ (87) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (69) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (18)* com/alipay/android/████████/█████/██████████████████████ (7) | android/support/annotation/NonNull -> androidx/annotation/NonNull (7)* com/alipay/android/████████/tab/█████████████████$2 (4) | android/support/v4/util/ArrayMap -> androidx/collection/ArrayMap (4)* com/alipay/android/████████/tab/█████████████████$1 (4) | android/support/v4/util/ArrayMap -> androidx/collection/ArrayMap (4)* com/alipay/mobile/██████/██████/████████████████████████████████$2$1 (7) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (7)* com/alipay/android/████/████████████$1 (5) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (5)* com/alipay/android/████████/████/███████████████$████████████████████$1$1 (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/mobile/████████/███/████████████$5 (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/mobile/████/████████/█████████████████████ (2) | android/support/v4/content/ContextCompat -> androidx/core/content/ContextCompat (2)* com/alipay/mobile/████████/███/████████████████████████$4 (9) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (9)* com/alipay/mobile/██████/██████/██████████████████████████ (306) | android/support/v4/app/FragmentActivity -> androidx/fragment/app/FragmentActivity (202) | android/support/v4/content/LocalBroadcastManager -> androidx/localbroadcastmanager/content/LocalBroadcastManager (22) | android/support/v4/app/FragmentManager -> androidx/fragment/app/FragmentManager (36) | android/support/v4/app/FragmentTransaction -> androidx/fragment/app/FragmentTransaction (38) | android/support/v4/app/Fragment -> androidx/fragment/app/Fragment (8)================= ====XML==== =================* res/layout/███████████████.xml (1)Reference detail: * res/layout/███████████████.xml (1) | android.support.v4.view.ViewPager -> androidx.viewpager.widget.ViewPager

基于apk进行androidx适配的能力,我们封装为了一个独立的jar工程,既可以集成到Gradle Plugin中使用,也可以发布为命令行工具使用。下面介绍下命令行的使用:
bx --input=src/test/resources/█████ .apk --output=build/intermediate

user % bx --help ___________ /|________/ /________(_)___/ / |/ / / /| | / __ / __/ ___/ __ / / __/|/ / ___ |/ / / / /_/ / // /_/ / / /_/ //| /_/|_/_/ /_/__,_/_/____/_/__,_//_/|_|Translate apk to compat with AndroidX dependency. Usage: bx [OPTIONS] [ARGS]...Options: --help显示帮助信息 --input待处理apk路径, 示例: --input=input.apk --output处理后产物目录, 示例: --output=output-dir --clean清理中间生成物, 示例: --clean=true --verbose输出调试的日志, 示例: --verbose=true --jetify使用jetify处理, 示例: --jetifye=true

构建平台打包,根据包类型不同,所需要编译的工程模块,插件逻辑如覆盖率插桩等,都有差异。因此不同类型的构建整体耗时有差异。androidx的增量耗时主要源自于集中式的转义的模块体量,随着独立适配模块数量的增加,集中耗时将逐步减少,根据数据采集分析,目前做一次全量的转义消耗时间比较稳定。最终理想情况是总包的构建基本不增加额外耗时。
将bx运用到钱包的构建插件后,可基于portal基线做批量的bundle兼容。开启兼容能力后,对整体构建时间有一定增量,通过持续优化可以逐步减少增量耗时。
当支付宝|当支付宝 App 遇见 AndroidX......
文章图片

原生的适配 Google本身提供了一天真对aar的适配方案,这部分可以直接上手。该套方案有三种使用方式:
  • 一是通过AS的Migrtate AndroidX,这要求AGP版本为3.2.0或更高;
  • 二是通过命令行,单独执行jar应用程序,这与上下文无关;
  • 三是源码改造,参考AGP的内部实现,去联动Jetify模块。
./jetifier-standalone -i-o

关于Jetify的介绍,这里摘录了一段文档说明:
Jetifier 会迁移指向 android.support. 软件包的 Java、XML、POM 和 ProGuard 引用,更改它们以使其指向相应的 androidx. 软件包。
由于 android.support. 的 ProGuard 通配符并不总是直接映射到 androidx. 软件包,因此 Jetifier 会生成所有符合条件的替代项。
如果 android.support.* 软件包中的某个类型不是来自任何支持库工件,那么只要该类型存在相关的映射,Jetifier 就会迁移该类型。但是,不能保证此迁移一定可行,因为可能没有足够通用的映射规则来涵盖所有自定义类型。
这个工具甚至可以反向执行,把androidx产物转位supportg产物,感兴趣可以试一下,-r参数。
在AS中,我们开启gradle参数配置后可以激活jetify机制,这个机制会对dependency依赖进行预处理,将依赖了support扩展库的aar编译为androidx版本。
android.useAndroidX=true android.enableJetifier=true

具体到代码实现,则是利用了Gradle的Artifact Transform机制,对依赖的aar文件拆解处理。相关源码可以参考:com.android.build.gradle.internal.dependency.JetifyTransform
这个transform中,最终会调用前面提到的jetify工具,Google将工具做成了独立的maven依赖:
androidx.jetifier:jetifier-core:1.0.0-beta10
https://developer.android.com...
后续 目前支付宝App已经完成的大部分业务的适配工作,部分有侵入性使用Support扩展库,如反射实例化View的需要业务进行适配。方案灰度也在持续推进当中。
除了AndroidX之外,我们也在推进Kotlin,Java8等在钱包构建体系的落地,期待钱包在完成AndroidX升级之后,全业务可以向原生化,现代Android开发更近一步。
当支付宝|当支付宝 App 遇见 AndroidX......
文章图片

欢迎也在积极拥抱AndroidX的小伙伴们,一起留言交流,共同探讨适配AndroidX的经验/教训。
关注我们,每周 3 篇移动技术实践&干货给你思考!

    推荐阅读