4

在我当前的 Kotlin Android 项目中,我只需要面对以下问题:

对于此类层次结构

abstract class MyClass {
...
}

class MySubClassA: MyClass() {
...
}

class MySubClassB: MyClass() {
...
}

class MySubClassC: MyClass() {
...
}

我写了一个这样的创建者函数:

private fun <T : MyClass> createMyClass(myClassType: KClass<T>): T? {
    val constructor = myClassType.primaryConstructor

    return constructor?.call()?.apply {
    ...
    }
}

这对于调试版本非常有效,但对于打开 R8 混淆的发布版本,构造函数总是null. 我对 R8 的设置是这样的:

-keep class kotlin.** { *; }
-keep class kotlin.Metadata { *; }
-dontwarn kotlin.**
-keepclassmembers class **$WhenMappings {
    <fields>;
}
-keepclassmembers class kotlin.Metadata {
    public <methods>;
}

-assumenosideeffects class kotlin.jvm.internal.Intrinsics {
    static void checkParameterIsNotNull(java.lang.Object, java.lang.String);
}

-keep class kotlin.reflect.jvm.internal.impl.builtins.BuiltInsLoaderImpl

在对 Kotlin 反射 API 进行了一些逆向工程之后,我想出了以下解决方法:

private fun <T : MyClass> createMyClass(myClassType: KClass<T>): T? {
    val constructor = myClassType.constructors.firstOrNull() {
        it.valueParameters.isEmpty()
    }

    return constructor?.call()?.apply {
    ...
    }
}

这似乎表明,类的构造函数信息由 R8 保留,但有关作为主构造函数的元信息会丢失。

有谁知道一般如何解决这个问题?

4

1 回答 1

0

我遇到了这个问题,并且能够使用@Keep注释解决它。

abstract class MyClass {
}

class MySubClassA @Keep constructor(): MyClass() {
}

class MySubClassB @Keep constructor(): MyClass() {
}

class MySubClassC @Keep constructor(): MyClass() {
}
于 2020-07-10T16:07:22.990 回答