2

I have the following Delegate...

fun integerPref(initialValue: Int) = object : ObservableProperty<Int>(initialValue) {

    override fun afterChange(property: KProperty<*>, oldValue: Int, newValue: Int) {
        getSharedPreference(INTEGER_PREF, EasyInjection.mode)
                .edit()
                .putInt(property.toString(), newValue)
                .apply()
    }

    override fun getValue(thisRef: Any?, property: KProperty<*>): Int {
        return getSharedPreference(INTEGER_PREF, EasyInjection.mode)
                .getInt(property.toString(), initialValue)
    }
}

This delegate aim to handling share preference. The problem happen when my application become MultiDex. When I use it like below code in activity, application crash.

class IntroduceActivity : BaseActivity() {

    private val CURRENT_VERSION = 1
    private val introVersion by integerPref()

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_introduce)

        if (introVersion == CURRENT_VERSION) {
            finish()
        }
   }
}

But, if I change the code to this one. Application not crash

class IntroduceActivity : BaseActivity() {

    private val CURRENT_VERSION = 1

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_introduce)

        var introVersion by integerPref()
        if (introVersion == CURRENT_VERSION) {
            finish()
        }
    }
}

I have read this document and it have some notice..

https://developer.android.com/studio/build/multidex.html

Caution: Do not execute MultiDex.install() or any other code through reflection or JNI before MultiDex.install() is complete. Multidex tracing will not follow those calls, causing ClassNotFoundException or verify errors due to a bad class partition between DEX files.

In conclusion, I want to know what happen in this scenario.

  • application crash happen on Android below 21
  • kotlin delegate use reflection
  • why change from field variable to local variable is solved problem ?
  • Logcat is not help

Logcat

FATAL EXCEPTION: main
java.lang.RuntimeException: Unable to start activity ComponentInfo{com.apg.mobile.som.uat/com.apg.mobile.som.activity.MainActivity}: java.lang.TypeNotPresentException: Type android/support/v4/app/ActivityCompatApi23$SharedElementCallbackImpl not present
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2180)
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2230)
    at android.app.ActivityThread.access$600(ActivityThread.java:141)
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1234)
    at android.os.Handler.dispatchMessage(Handler.java:99)
    at android.os.Looper.loop(Looper.java:137)
    at android.app.ActivityThread.main(ActivityThread.java:5041)
    at java.lang.reflect.Method.invokeNative(Native Method)
    at java.lang.reflect.Method.invoke(Method.java:511)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560)
    at dalvik.system.NativeStart.main(Native Method)
 Caused by: java.lang.TypeNotPresentException: Type android/support/v4/app/ActivityCompatApi23$SharedElementCallbackImpl not present
    at java.lang.Class.getDeclaredClasses(Native Method)
    at java.lang.Class.getDeclaredClasses(Class.java:534)
    at kotlin.reflect.jvm.internal.impl.load.java.structure.reflect.ReflectJavaClass.getInnerClassNames(ReflectJavaClass.kt:35)
    at kotlin.reflect.jvm.internal.impl.load.java.structure.reflect.ReflectJavaClass.getInnerClassNames(ReflectJavaClass.kt:27)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.descriptors.LazyJavaClassMemberScope$nestedClassIndex$1.invoke(LazyJavaClassMemberScope.kt:617)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.descriptors.LazyJavaClassMemberScope$nestedClassIndex$1.invoke(LazyJavaClassMemberScope.kt:64)
    at kotlin.reflect.jvm.internal.impl.storage.LockBasedStorageManager$LockBasedLazyValue.invoke(LockBasedStorageManager.java:323)
    at kotlin.reflect.jvm.internal.impl.storage.LockBasedStorageManager$LockBasedNotNullLazyValue.invoke(LockBasedStorageManager.java:370)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.descriptors.LazyJavaClassMemberScope$nestedClasses$1.invoke(LazyJavaClassMemberScope.kt:626)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.descriptors.LazyJavaClassMemberScope$nestedClasses$1.invoke(LazyJavaClassMemberScope.kt:64)
    at kotlin.reflect.jvm.internal.impl.storage.LockBasedStorageManager$MapBasedMemoizedFunction.invoke(LockBasedStorageManager.java:408)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.descriptors.LazyJavaClassMemberScope.getContributedClassifier(LazyJavaClassMemberScope.kt:651)
    at kotlin.reflect.jvm.internal.impl.resolve.scopes.InnerClassesScopeWrapper.getContributedClassifier(InnerClassesScopeWrapper.kt:29)
    at kotlin.reflect.jvm.internal.impl.resolve.jvm.JavaDescriptorResolver.resolveClass(JavaDescriptorResolver.kt:38)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.SingleModuleClassResolver.resolveClass(ModuleClassResolver.kt:29)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.types.JavaTypeResolver.computeTypeConstructor(JavaTypeResolver.kt:132)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.types.JavaTypeResolver.computeSimpleJavaClassifierType(JavaTypeResolver.kt:114)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.types.JavaTypeResolver.transformJavaClassifierType(JavaTypeResolver.kt:90)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.types.JavaTypeResolver.transformJavaType(JavaTypeResolver.kt:52)
    at kotlin.reflect.jvm.internal.impl.load.java.lazy.descriptors.LazyJavaClassDescriptor$LazyJavaClassTypeConstructor.computeSupertypes(LazyJavaClassDescriptor.kt:178)
    at kotlin.reflect.jvm.internal.impl.types.AbstractTypeConstructor$supertypes$1.invoke(AbstractTypeConstructor.kt:35)
    at kotlin.reflect.jvm.internal.impl.types.AbstractTypeConstructor$supertypes$1.invoke(AbstractTypeConstructor.kt:22)
    at kotlin.reflect.jvm.internal.impl.storage.LockBasedStorageManager$LockBasedLazyValue.invoke(LockBasedStorageManager.java:
tynn
  • 38,113
  • 8
  • 108
  • 143
Jongz Puangput
  • 5,527
  • 10
  • 58
  • 96

0 Answers0