如何从AAR依赖中排除特定资源?

Com*_*are 30 android android-gradle-plugin

模块的build.gradle文件是否有一种相当简单的方法来指示应该排除来自依赖项的某些文件?我特别想从AAR中排除某些资源.


LeakCanary是一个有趣的库,可帮助追踪内存泄漏.但是,它的无证要求compileSdkVersion是21或更高.虽然大多数项目都不应该有这个问题,但是图书馆compileSdkVersion没有充分理由要求某个项目是不合适的.开发团队可能已将其compileSdkVersion作为一般策略的一部分冻结,仅将这些设置更改为其应用程序或其他内容的主要版本更新的一部分.

在这种情况下,对于LeakCanary的v1.3.1至少,唯一的原因compileSdkVersion是AFAICT,因为AAR有一个res/values-v21/目录,包含一个继承自的主题定义Theme.Material.诊断活动使用此主题.最终用户从未见过该活动,只有debug构建中的开发人员才能看到.坦率地说,主题方面的活动看起来并不重要.迫使compileSdkVersion21只是让那个诊断活动有一个特定的主题是,恕我直言,愚蠢.

如果作为compile指令的一部分,我们可以说"嘿,请跳过res/values-v21/这个AAR,m'kay,这很好." 由于-v21主题只是提供了其他地方定义的主题的替代定义,因此删除-v21主题不会破坏构建或在运行时中断事物,而只会给我们一个Holo主题诊断活动.

我没有看到这个答案如何与依赖关系一起工作.我也不确定它是否完整,它似乎似乎没有得到支持.它也没有真正符合"简单"的条件 - 我不希望有人尝试将其放在build.gradle文件中只是为了阻止像LeakCanary这样的诊断库中的单个文件.

那么,是否有一些比这更简单的东西适用于当前版本的Android插件Gradle?

Ily*_*lya 15

编辑:

为您写了高级gradle任务:

final List<String> exclusions = [];

Dependency.metaClass.exclude = { String[] currentExclusions ->
    currentExclusions.each {
        exclusions.add("${getGroup()}/${getName()}/${getVersion()}/${it}")
    }
    return thisObject
}

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    testCompile 'junit:junit:4.12'
    compile ('com.android.support:appcompat-v7:20.+')
    debugCompile ('com.squareup.leakcanary:leakcanary-android:1.3.1')
            .exclude("res/values-v21/values-v21.xml")
    releaseCompile ('com.squareup.leakcanary:leakcanary-android-no-op:1.3.1')
}

tasks.create("excludeTask") << {
    exclusions.each {
        File file = file("${buildDir}/intermediates/exploded-aar/${it}")
        println("Excluding file " + file)
        if (file.exists()) {
            file.delete();
        }
    }
}

tasks.whenTaskAdded({
    if (it.name.matches(/^process.*Resources$/)) {
        it.dependsOn excludeTask
    }
})
Run Code Online (Sandbox Code Playgroud)

现在,您可以.exclude()对每个依赖项使用方法,提供到路径列表中,您希望从指定的依赖项中排除.此外,您可以堆叠.exclude()方法调用.

  • 太糟糕了,它在 2020 年不再适用于 gradle 4.0.1 (2认同)

And*_*wJC 5

我相信您可以使用Android Gradle Plugin DSL 的PackagingOptions工具更优雅地解决这个问题。

我自己可以使用它来排除一些我不需要在项目中由 AAR 引入的本机库。

android {
    ...
    packagingOptions {
        exclude '/lib/armeabi-v7a/<file_to_exclude>'
    }
}
Run Code Online (Sandbox Code Playgroud)

对于问题中概述的情况,我相信这会起作用:

android {
    ...
    packagingOptions {
        exclude '/res/values-v21/<file_to_exclude>'
    }
}
Run Code Online (Sandbox Code Playgroud)

  • 有时我会尝试一下,但我怀疑这是否有效。首先,“pacakgingOptions”在构建过程中太晚了——构建应该会因“compileSdkVersion”而失败。其次,据我所知,“packagingOptions”是指根据 APK 中的位置排除某些内容,并且 XML 资源不会像普通文件一样进入 APK。不过谢谢! (3认同)