29

我们一直在使用这样的片段来重命名 Gradle 构建生成的 APK 文件:

android.applicationVariants.all { variant ->
    variant.outputs.all {
        outputFileName = "${variant.name}-${variant.versionName}.apk"
    }
}

来源:https ://developer.android.com/studio/build/gradle-plugin-3-0-0-migration#variant_output

我现在正在将我的转换build.gradlebuild.gradle.kts,即转换为 Gradle Kotlin DSL。这是最后丢失的部分之一:我不知道如何访问outputFileName.

根据 API 文档,它甚至似乎不存在:

  • BaseVariant.getOutputs()返回DomainObjectCollection<BaseVariantOutput>提供all片段中使用的方法的 a。
  • BaseVariantOutputextends OutputFilewhich extendsVariantOutput但这些都没有一个outputFileName或任何匹配名称的 getter 或 setter。

所以,我怀疑有一些高级的 Groovy 魔法在起作用——但是我如何在 Kotlin 中实现呢?

4

3 回答 3

31

@david.mihola 答案的一个简化版本:

android {

    /**
    * Notes Impl: Use DomainObjectCollection#all
    */
    applicationVariants.all {
        val variant = this
        variant.outputs
            .map { it as com.android.build.gradle.internal.api.BaseVariantOutputImpl }
            .forEach { output ->
                val outputFileName = "YourAppName - ${variant.baseName} - ${variant.versionName} ${variant.versionCode}.apk"
                println("OutputFileName: $outputFileName")
                output.outputFileName = outputFileName
            }
    }

}
于 2019-09-20T23:04:22.683 回答
28

浏览 Android Gradle 插件的源代码,我想我找到了答案——我们开始吧:

我们实际上是在处理类型的对象,BaseVariantOutputImpl而这个类确实有这两种方法:

public String getOutputFileName() {
    return apkData.getOutputFileName();
}

public void setOutputFileName(String outputFileName) {
    if (new File(outputFileName).isAbsolute()) {
        throw new GradleException("Absolute path are not supported when setting " +
                    "an output file name");
    }
    apkData.setOutputFileName(outputFileName);
}

利用这些知识,我们现在可以:

import com.android.build.gradle.internal.api.BaseVariantOutputImpl

然后像这样投射我们的目标对象:

applicationVariants.all(object : Action<ApplicationVariant> {
    override fun execute(variant: ApplicationVariant) {
        println("variant: ${variant}")
        variant.outputs.all(object : Action<BaseVariantOutput> {
            override fun execute(output: BaseVariantOutput) {

                val outputImpl = output as BaseVariantOutputImpl
                val fileName = output.outputFileName
                        .replace("-release", "-release-v${defaultConfig.versionName}-vc${defaultConfig.versionCode}-$gitHash")
                        .replace("-debug", "-debug-v${defaultConfig.versionName}-vc${defaultConfig.versionCode}-$gitHash")
                println("output file name: ${fileName}")
                outputImpl.outputFileName = fileName
            }
        })
    }
})

所以,我猜:是的,Groovy 有一些魔力在起作用,即 Groovy 的动态类型系统允许您从代码中访问getOutputFileNamesetOutputFileName(通过缩写outputImpl.outputFileName语法,如 Kotlin),希望它们在运行时存在,即使您知道的编译时接口没有它们。

于 2018-06-11T16:11:57.493 回答
8

使用 lambda 的较短版本:

    applicationVariants.all{
        outputs.all {
            if(name.contains("release"))
                (this as BaseVariantOutputImpl).outputFileName = "../../apk/$name-$versionName.apk"
        }
    }

这会将 APK 放入 app/apk 文件夹,名称由变体名称和版本代码组成。
您可以根据需要更改文件名的格式。
重要提示:它必须只在发布版本上完成,因为路径中的“..”会破坏调试构建过程并出现奇怪的错误。

于 2020-03-25T12:42:04.953 回答