Android Espresso 内部匕首工厂类的 NoClassDefFoundError

Posted

技术标签:

【中文标题】Android Espresso 内部匕首工厂类的 NoClassDefFoundError【英文标题】:NoClassDefFoundError for Android Espresso internal dagger factory class 【发布时间】:2016-10-01 20:30:34 【问题描述】:

我正在尝试将 Espresso 测试框架添加到我的项目中。但是,我被这个 NoClassDefFoundError 困住了 3 天。在 Google 上搜索后,我发现测试 APK 默认使用的是 DEBUG 构建类型。出于某种原因,我需要为调试构建类型设置 Proguard。但是,在运行测试配置时出现以下错误:

06-02 15:27:01.105 19436-19457/com.lingyue.Yqgandroid E/TestLoader: Could not find class: android.support.test.espresso.base.UiControllerModule_ProvideUiControllerFactory
06-02 15:27:01.107 19436-19457/com.lingyue.YqgAndroid I/art: Rejecting re-init on previously-failed class java.lang.Class<android.support.test.espresso.core.deps.dagger.internal.Factory>
06-02 15:27:01.107 19436-19457/com.lingyue.YqgAndroid I/art: Rejecting re-init on previously-failed class java.lang.Class<android.support.test.espresso.base.ViewFinderImpl_Factory>
06-02 15:27:01.107 19436-19457/com.lingyue.YqgAndroid I/art: Rejecting re-init on previously-failed class java.lang.Class<android.support.test.espresso.base.ViewFinderImpl_Factory>
06-02 15:27:01.109 19436-19457/com.lingyue.YqgAndroid E/TestLoader: Could not find class: android.support.test.espresso.base.ViewFinderImpl_Factory
06-02 15:27:01.114 19436-19457/com.lingyue.YqgAndroid I/art: Rejecting re-init on previously-failed class java.lang.Class<android.support.test.espresso.core.deps.dagger.internal.Factory>
06-02 15:27:01.115 19436-19457/com.lingyue.YqgAndroid E/AndroidRuntime: FATAL EXCEPTION: Instr: android.support.test.runner.AndroidJUnitRunner
    Process: com.lingyue.YqgAndroid, PID: 19436
    java.lang.NoClassDefFoundError: android.support.test.espresso.core.deps.dagger.internal.Factory
    at java.lang.Class.classForName(Native Method)
    at java.lang.Class.forName(Class.java:324)
    at android.support.test.internal.runner.TestLoader.doLoadClass(TestLoader.java:92)
    at android.support.test.internal.runner.TestLoader.loadIfTest(TestLoader.java:113)
    at android.support.test.internal.runner.TestRequestBuilder.loadClassesFromClassPath(TestRequestBuilder.java:801)
    at android.support.test.internal.runner.TestRequestBuilder.build(TestRequestBuilder.java:747)
    at android.support.test.runner.AndroidJUnitRunner.buildRequest(AndroidJUnitRunner.java:354)
    at android.support.test.runner.AndroidJUnitRunner.onStart(AndroidJUnitRunner.java:260)
    at android.app.Instrumentation$InstrumentationThread.run(Instrumentation.java:1879)

我的 build.gradle 对于依赖项看起来像这样:

dependencies 
    compile "com.android.support:cardview-v7:$supportVersion"
    compile "com.android.support:recyclerview-v7:$supportVersion"
    compile 'com.nostra13.universalimageloader:universal-image-loader:1.9.3'
    compile 'com.loopj.android:android-async-http:1.4.9'
    compile 'com.qiniu:qiniu-android-sdk:7.0.9'
    compile 'com.mcxiaoke.gradle:packer-helper:1.0.4'
    compile 'me.henrytao:smooth-app-bar-layout:23.2.1.1'
    compile 'com.alipay.euler:andfix:0.4.0@aar'
    compile 'com.umeng.analytics:analytics:6.0.0'
    compile fileTree('libs')
    compile project(':framework')
    compile project(':yqdsdk')

    // Only needed at compilation
    provided 'com.google.dagger:dagger-compiler:2.0'
    provided 'org.glassfish:javax.annotation:10.0-b28'

    // For testing
    androidTestCompile "com.android.support:support-annotations:$supportVersion"
    androidTestCompile 'com.android.support.test:runner:0.5'
    androidTestCompile 'com.android.support.test:rules:0.5'
    androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2') 
        exclude group: 'javax.inject'
    

构建类型如下所示:

buildTypes 
    debug 
        minifyEnabled true
        proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        testProguardFile 'test-proguard-rules.pro'
    

    release 
        minifyEnabled true
        proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
    

testProguardFile.pro 看起来像这样:

-dontobfuscate
-dontwarn

我做了什么:

    如果我通过设置 minifyEnabled=false 将 Proguard 关闭 以进行调试构建,则测试会成功运行。否则我会收到上述错误。 我尝试将 testProguardFile 放在“defaultConfig”和“debug”下,但都无济于事。 我尝试在关闭 Proguard 的情况下再添加一种构建类型“uiTest”,但在将 testBuildType 设置为“uiTest”后,测试配置将不起作用。错误是 Android Studio 无法识别测试运行器 AndroidJUnitRunner。

在我看来,问题在于 proguard 和 dagger2 的混合使用,但我的想法已经不多了。请帮忙。

最好的祝福

【问题讨论】:

嗨,我遇到了完全相同的问题。尝试在启用 ProGuard 的情况下运行我的仪器测试,但测试因该异常而崩溃。我还有一个带有类似规则的单独测试 ProGuard 文件。 Nicholas,您最终找到解决此问题的方法了吗? 我遇到了similar issue,但在我的案例中,测试在本地设备上通过,但在 Firebase 测试实验室失败。 【参考方案1】:

在做了更多研究后,我找到了解决这个问题的方法,虽然这不是直截了当的:

    添加新的构建类型并同步构建项目,例如

    uiTest 缩小启用真

    在 Android Studio 的左下方,单击“构建变体”。对于您的应用程序模块,选择新添加的构建变体,例如“UiTest”。

    运行应用程序。

这有点不方便,因为当您需要在不同的构建变体下进行测试时,您需要来回切换。但它让 Expresso 运行起来。

【讨论】:

【参考方案2】:

我有一个similar problem 并且能够通过将以下规则添加到 proguard-rules.pro 来修复它:

-keep class javax.inject.**  *; 

我不完全确定您的问题是否相同。 另外我不确定您是否需要从 espresso-core 依赖项中排除 javax.inject 组。

【讨论】:

以上是关于Android Espresso 内部匕首工厂类的 NoClassDefFoundError的主要内容,如果未能解决你的问题,请参考以下文章

内部嵌套类的 Kotlin 工厂

无法实例化应用程序匕首android

匕首可以使android活动创建速度变慢吗?

未解决的参考匕首 2 + kotlin + android gradle

Android的报错提示:Failed to resolve: com.android.support.test.espresso:espresso-core:3.0.2

使用匕首 2 在 Kotlin 中的 AppWidgetProvider 中的字段注入