More than one file was found with OS independent path 'META-INF/library_release.kotlin_module'
最近在引入两个kotlin写的aar库时编译器报了这样的一个错误。说是打包时存在两个相同的文件,文件路径是META-INF/library_release.kotlin_module
。
这个文件是干什么用的呢?
这就要从 kotlin 的package-level-functions
说起了。
我们都知道 kotlin 支持文件级别的函数、变量申明,也就是说我们可以直接在一个文件中申明变量、方法,而不像Java,必须有一个类来承接这些定义。
每个这种有了文件级别的申明的 kotlin source file,都会默认创建一个同样命名,但首字母大写,末尾加了“Kt”后缀的class类。
比如FileExt.kt
会编译成FileExtKt.class
在Java代码中,可以通过编译后的类名访问 kotlin 申明在顶层的函数和变量。因此,在同一包名下,不支持两个命名相同的kotlin source文件。
但是,我们通过@file:JvmName("CustomName")
可以显式的指定一个文件编译后的类名,或者通过@file:JvmMultifileClass
指定哪几个文件共享一个类名。这就是通过注解告诉kotlin编译器,我这个文件要编译到哪个生成类中。
为了实现这个功能,kotlin专门引入了一个resource文件,用来指导kotlin的编译。
没错,这个文件就是 META-INF/<module_name>.kotlin_module
.
网上有很多人解决上面的问题时,将 .kotlin_module
文件在打包流程中exclude,实际上这种做法是不正确的。
其实只需要保证module名字没有冲突,生成的.kotlin_module
文件自然也就不会冲突。
在 gradle 工程中,这个文件的名字由当前工程的module
名字 + build variant
组成。
例如在一个 aar sdk 中,你可能会创建一个叫做 library
的module 文件夹,release
发布出去后,生成的aar中会带有'META-INF/library_release.kotlin_module'
这个文件。
如果其他的工程也使用了相同的module name
,比如另外一个aar刚好也用了library
作为module名称,就会报上面的错误。
通过kotlinOptions可以指定moduleName
名称。比如下面的这段代码会将moduleName
修改为项目的group.artifact
名字。
android {
compileSdkVersion 25
buildToolsVersion "25.0.0"
compileOptions {
kotlinOptions.freeCompilerArgs += ['-module-name', "$PUBLISH_GROUP_ID.$PUBLISH_ARTIFACT_ID"]
}
defaultConfig {
minSdkVersion 8
targetSdkVersion 25
versionCode 1
versionName "1.0.5"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
}
注意:PUBLISH_GROUP_ID 和 PUBLISH_ARTIFACT_ID 是自定义的变量,你可以将它改为自己的项目中的具体值。
参考文章:
https://github.com/Takhion/android-extras-delegates/issues/1
https://blog.jetbrains.com/kotlin/2015/09/kotlin-m13-is-out/
http://kotlinlang.org/docs/reference/java-interop.html#package-level-functions