为啥我得到 kotlin.UninitializedPropertyAccessException 即使 lateinit 属性已初始化(可能)

Posted

技术标签:

【中文标题】为啥我得到 kotlin.UninitializedPropertyAccessException 即使 lateinit 属性已初始化(可能)【英文标题】:Why am i getting kotlin.UninitializedPropertyAccessException even if lateinit property is initialized (probably)为什么我得到 kotlin.UninitializedPropertyAccessException 即使 lateinit 属性已初始化(可能) 【发布时间】:2020-08-15 15:42:22 【问题描述】:

好的,所以我在小部件中声明了一个 lateinit var job,如下所示。

class TempHumidDisplayWidget : AppWidgetProvider(), CoroutineScope 
    private lateinit var job: Job
    override val coroutineContext: CoroutineContext
        get() = Dispatchers.Main + job
    ...

在onEnabled函数中,我对其进行了初始化:

override fun onEnabled(context: Context) 
    job = Job()
    Log.i("com.github.animeshz", "On enabled")
    ...

在 onUpdate 函数中,我将使用 launch 调用 coroutineContext 的 get 函数来调度使用作业变量的协程。

override fun onUpdate(
    context: Context,
    appWidgetManager: AppWidgetManager,
    appWidgetIds: IntArray
) 
    // There may be multiple widgets active, so update all of them
    for (appWidgetId in appWidgetIds) 
        launch  updateAppWidget(context, appWidgetManager, appWidgetId) 
    

在 logcat 中,我收到以下错误:

4425-4425/? I/com.github.animeshz: On enabled
...
2020-05-01 11:06:06.639 4425-4425/? E/androidRuntime: FATAL EXCEPTION: main
    Process: com.github.animeshz.shivamwidget, PID: 4425
    java.lang.RuntimeException: Unable to start receiver com.github.animeshz.shivamwidget.TempHumidDisplayWidget: kotlin.UninitializedPropertyAccessException: lateinit property job has not been initialized
        at android.app.ActivityThread.handleReceiver(ActivityThread.java:3183)
        at android.app.ActivityThread.-wrap18(ActivityThread.java)
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1636)
        at android.os.Handler.dispatchMessage(Handler.java:102)
        at android.os.Looper.loop(Looper.java:154)
        at android.app.ActivityThread.main(ActivityThread.java:6334)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:886)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:776)
     Caused by: kotlin.UninitializedPropertyAccessException: lateinit property job has not been initialized
        at com.github.animeshz.shivamwidget.TempHumidDisplayWidget.getCoroutineContext(TempHumidDisplayWidget.kt:26)
        at kotlinx.coroutines.CoroutineContextKt.newCoroutineContext(CoroutineContext.kt:33)
        at kotlinx.coroutines.BuildersKt__Builders_commonKt.launch(Builders.common.kt:50)
        at kotlinx.coroutines.BuildersKt.launch(Unknown Source)
        at kotlinx.coroutines.BuildersKt__Builders_commonKt.launch$default(Builders.common.kt:47)
        at kotlinx.coroutines.BuildersKt.launch$default(Unknown Source)
        at com.github.animeshz.shivamwidget.TempHumidDisplayWidget.onUpdate(TempHumidDisplayWidget.kt:38)
        at android.appwidget.AppWidgetProvider.onReceive(AppWidgetProvider.java:66)
        at android.app.ActivityThread.handleReceiver(ActivityThread.java:3171)
        at android.app.ActivityThread.-wrap18(ActivityThread.java) 
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1636) 
        at android.os.Handler.dispatchMessage(Handler.java:102) 
        at android.os.Looper.loop(Looper.java:154) 
        at android.app.ActivityThread.main(ActivityThread.java:6334) 
        at java.lang.reflect.Method.invoke(Native Method) 
        at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:886) 
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:776) 

很明显onEnabled 是在onUpdate 方法之前调用的,但是job 变量没有以某种方式初始化,我不知道如何修复或进一步调试它

怎么可能onEnabled中初始化的job变量实际上没有在onUpdate函数中初始化?这是一个错误还是什么?

任何帮助将不胜感激,在此先感谢!

【问题讨论】:

看起来不错。也许onEnabledonUpdate 在不同的实例上被调用?尝试在日志中包含hashCode(或System.identityHashCode(this))docs.oracle.com/javase/10/docs/api/java/lang/… 另外,如果job 不依赖于context 参数,为什么它根本需要lateinit @AlexeyRomanov 是的,你是对的,有不同的小部件类实例。但如果是这种情况,那么当用户使用小部件时我将如何创建对象并在用户删除小部件时销毁。 它需要延迟初始化,因为我想将作业作为生命周期附加,以便在用户删除小部件时可以取消正在进行的任务。因此在onEnabled 上创建了一个工作,并在onDisabled 上销毁了一个工作 【参考方案1】:

这只是吐槽,因为我不知道 Android 中小部件的详细信息,但这样的事情可能会起作用:

class TempHumidDisplayWidget : AppWidgetProvider() 
    companion object 
        val jobs = mutableMapOf<Context, Job>()

        fun getScope(context: Context): CoroutineScope 
            val job = jobs[context] ?: throw IllegalStateException("Context $context not enabled currently")
            return object : CoroutineScope  
                override val coroutineContext: CoroutineContext = Dispatchers.Main + job
                ...
            
        
    

    override fun onEnabled(context: Context) 
        jobs[context] = Job()
        ...
    

    override fun onDisabled(context: Context) 
        jobs.remove(context)?.cancel()
        ...
    

    override fun onUpdate(
        context: Context,
        appWidgetManager: AppWidgetManager,
        appWidgetIds: IntArray
    ) 
        val scope = getScope(context)
        for (appWidgetId in appWidgetIds) 
            scope.launch  updateAppWidget(context, appWidgetManager, appWidgetId) 
        
    

“静态地”保留对Contexts 的引用通常是个坏主意,但这可能没问题,因为它们已在onDisabled 中删除。或者也许改用WeakHashMap

【讨论】:

以上是关于为啥我得到 kotlin.UninitializedPropertyAccessException 即使 lateinit 属性已初始化(可能)的主要内容,如果未能解决你的问题,请参考以下文章

为啥我得到一个 nullPointerException? [复制]

为啥我没有得到任何回应

为啥我得到黑色纹理?

为啥我得到分段(核心转储)?

为啥我得到 getApplicationcontext() null?

为啥我不能从数组中得到正确的日期?