前段时间做一个需求,因为要做复杂的动画,所以选择了Airbnb的lottie动画框架。但在引入lottie的过程中碰到了一个重复类(duplicate class)的编译错误,解决这个问题花了我一些时间,这里将其归纳为一类问题,并说明一下解决的方法。
我新建一个Android应用来重现这个问题(操作系统为macOS 10.13.6):
1、用Android Studio(3.4.2)创建一个新的工程,minSdkVersion为16
2、编译、安装成功,运行后界面上正常显示Hello World
3、在app module的build.gradle文件中用implementation 'com.airbnb.android:lottie:3.0.1'引入版本号为3.0.1的lottie库
4、编译
编译时报如下错误:
ERROR: Manifest merger failed : Attribute application@appComponentFactory value=(android.support.v4.app.CoreComponentFactory) from [com.android.support:support-compat:28.0.0] AndroidManifest.xml:22:18-91
is also present at [androidx.core:core:1.0.0] AndroidManifest.xml:22:18-86 value=(androidx.core.app.CoreComponentFactory).
Suggestion: add 'tools:replace="android:appComponentFactory"' to
错误截图:
根据错误的描述,我们知道这是在merge主工程和依赖库的Manifest文件时出错了。主工程通过application@appComponentFactory属性指定用support库(com.android.support:support-compat:28.0.0)里的CoreComponentFactory类,而有个依赖库(并没有指出是哪个依赖库)通过application@appComponentFactory属性指定用androidx库(androidx.core:core:1.0.0)里的CoreComponentFactory类,这时产生了冲突,便没法merge了。按照提示,我们在主工程的Manifest文件的
tools:replace="android:appComponentFactory"
android:appComponentFactory="android.support.v4.app.CoreComponentFactory"
再编译,不报之前的错误了,但报了一个新的错误:
Caused by: com.android.ide.common.workers.WorkerExecutorException: 1 exception was raised by workers:
java.lang.RuntimeException: Duplicate class android.support.v4.app.INotificationSideChannel found in modules classes.jar (androidx.core:core:1.0.0) and classes.jar (com.android.support:support-compat:28.0.0)
看描述是因为androidx包(androidx.core:core:1.0.0)和support包(com.android.support:support-compat:28.0.0)中都有INotificationSideChannel这个类,冲突了。因为主工程用了28.0.0的support包,所以是依赖库使用了1.0.0的androidx.core包导致的。到底是哪个依赖库错误信息里并没有说。
我们可以在Android Studio的终端(Terminal)中用如下命令打印出工程所有的依赖库:
./gradlew app:dependencies
因为打印出的东西非常多,我们在Terminal中用command + f 搜索包名androidx.core:
可以看到,是lottie这个依赖库使用了androidx.core。知道了是谁引入了这个重复类,接下来就看怎么解决这个问题。
要解决这个问题,我们首先得知道AndroidX是什么。下面是官方文档的截图:
网址:https://developer.android.com/jetpack/androidx
可以看到,AndroidX完全取代了管理混乱的support库,并且以后所有新的support库都将放到AndroidX中。
到现在,解决问题的2个方法就出来了:
1、将主工程的依赖从support库切到AndroidX库
2、将lottie依赖的AndroidX库切到support库
方法1有点麻烦,我选择了方法2。因为lottie稍老一点的版本并没有使用AndroidX库,所以我将lottie的版本降到2.7.0,再编译就不报错,可以正常使用了。