如何使用导航架构组件修复动态功能模块中片段的发布版本中的ClassNotFoundException?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了如何使用导航架构组件修复动态功能模块中片段的发布版本中的ClassNotFoundException?相关的知识,希望对你有一定的参考价值。

我正在尝试使用动态功能模块构建应用程序,而无需按需支持。

这就是我的app结构:

-app
---MainActivity
-base
---SharedFile
-comments
---CommentsFragment
-posts
---PostsFragment

当然,每个模块中还有其他文件。应用程序的调试版本工作正常,发布版本编译成功。但是,在发布版本中,我遇到了一个运行时崩溃,其中有一个用于CommentsFragment和PostsFragment的ClassNotFound Exception

即使没有在我的发布版本(minifyEnabled false)上启用proguard,也会发生此错误。当我启用Proguard时,结果是一样的。我尝试将以下Proguard规则添加到app模块proguard文件(取自here):

-keep public class * extends android.app.Activity
-keep public class * extends androidx.fragment.app.Fragment
-keep public class * extends android.app.Application
-keep public class * extends android.app.Service
-keep public class * extends android.content.BroadcastReceiver
-keep public class * extends android.content.ContentProvider

-keep public class * extends android.view.View {
      public <init>(android.content.Context);
      public <init>(android.content.Context, android.util.AttributeSet);
      public <init>(android.content.Context, android.util.AttributeSet, int);
      public void set*(...);
}

-keepclasseswithmembers class * {
    public <init>(android.content.Context, android.util.AttributeSet);
}

-keepclasseswithmembers class * {
    public <init>(android.content.Context, android.util.AttributeSet, int);
}

-keepclassmembers class * extends android.content.Context {
    public void *(android.view.View);
    public void *(android.view.MenuItem);
}

-keepclassmembers class * implements android.os.Parcelable {
    static ** CREATOR;
}

-keepclassmembers class **.R$* {
    public static <fields>;
}

-keepclassmembers class * {
    @android.webkit.javascriptInterface <methods>;
}

结果仍然相同:运行时崩溃。

这是我的app模块build.gradle文件:

apply plugin: 'com.android.application'
apply plugin: 'kotlin-android'
apply plugin: 'kotlin-android-extensions'
apply plugin: 'androidx.navigation.safeargs.kotlin'

android {
    compileSdkVersion ProjectProps.compileSdk
    defaultConfig {
        applicationId ProjectProps.applicationId
        minSdkVersion ProjectProps.minSdk
        targetSdkVersion ProjectProps.targetSdk
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
        }
    }
    compileOptions {
        sourceCompatibility = JavaVersion.VERSION_1_8
        targetCompatibility = JavaVersion.VERSION_1_8
    }

    dynamicFeatures = [":users", ":posts"]

    packagingOptions {
        exclude 'META-INF/atomicfu.kotlin_module'
    }

    dataBinding {
        enabled = true
    }
}

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation project(path: ':base')
    api project(path: ':json-placeholder-repository')

    ...
}

这是我得到的完整错误:

Caused by: java.lang.ClassNotFoundException: Didn't find class "com.app.usersmodule.UsersFragment" on path: DexPathList[[zip file "/data/app/com.app-r8BPSasOKhil086vurQf4g==/base.apk"],nativeLibraryDirectories=[/data/app/com.app-r8BPSasOKhil086vurQf4g==/lib/arm64, /system/lib64, /vendor/lib64]]
        at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:134)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:379)
        at java.lang.ClassLoader.loadClass(ClassLoader.java:312)
        at androidx.fragment.app.FragmentFactory.loadClass(FragmentFactory.java:50)
        at androidx.fragment.app.FragmentFactory.loadFragmentClass(FragmentFactory.java:91)

我不确定我做错了什么,因为Debug构建工作正常。

这似乎是Nav AAC的一个限制,因为它会在图表膨胀期间检查运行时是否存在所有类。我不确定为什么班级不存在。我在项目中禁用了Proguard,因此没有任何代码被删除。

答案

问题原来是将签名的APK部署到设备,而不是使用Android App Bundle格式生成签名的APK。

创建一个已签名的AAB,然后使用来自Google的bundletool将其转换为已签名的APK,会创建一个包含其中所有片段和类的APK。这适用于我的设备就好了。

如果您的项目具有动态功能模块,请使用AAB。

以上是关于如何使用导航架构组件修复动态功能模块中片段的发布版本中的ClassNotFoundException?的主要内容,如果未能解决你的问题,请参考以下文章

如何使用导航架构组件查找子片段

底部应用栏在使用片段导航时向上/向下滑动(导航架构组件)

如何使用新的导航架构组件从扩展 BroadcastReceiver 的类导航到片段

导航架构组件 - 对话框片段

使用导航架构组件添加(而不是替换)片段

在 FAB 单击时导航到片段(导航架构组件)