【问题标题】:Database query throws KotlinNullPointerException while trying get LiveData数据库查询在尝试获取 LiveData 时抛出 KotlinNullPointerException
【发布时间】:2020-06-26 11:32:48
【问题描述】:

我正在尝试在 kotlin/android 中创建房间数据库,但是当我尝试从数据库中获取 LiveData 时,我得到了KotlinNullPointerException。我有两个表CarRefueling,从一个片段中我可以毫无问题地获取Car 数据,但在第二个片段中,我在尝试获取Refueling LiveData 时出错。我找不到这两个片段/DAO 之间的任何区别。下面是一个简化的代码。

加油数据类:

@Entity(tableName = "refueling_table", foreignKeys = [ForeignKey(entity = Car::class,
    parentColumns = arrayOf("carID"),
    childColumns = arrayOf("carID"),
    onDelete = ForeignKey.CASCADE)]
)
data class Refueling(
    @PrimaryKey(autoGenerate = true)
    val refuelingID: Long = 0L,

    @ColumnInfo(name = "carID")
    val carID: Long = 0L,
)

加油DAO:

@Dao
interface RefuelingDAO
{
    @Query("SELECT * FROM refueling_table ORDER BY refuelingID DESC")
    fun getAll(): LiveData<List<Refueling>>
}

应用数据库:

@Database(entities = [Car::class, Refueling::class], version = 1, exportSchema = false)
abstract class AppDatabase : RoomDatabase()
{
    abstract val carDAO: CarDAO
    abstract val refuelingDAO: RefuelingDAO

    companion object
    {
        @Volatile
        private var INSTANCE: AppDatabase? = null

        fun getInstance(context: Context): AppDatabase
        {
            synchronized(this)
            {
                var instance = INSTANCE
                if (instance == null)
                {
                    instance = Room.databaseBuilder(
                        context.applicationContext,
                        AppDatabase::class.java,
                        "app_database"
                    ).fallbackToDestructiveMigration().build()
                    INSTANCE = instance
                }
                return instance
            }
        }
    }
}

在片段中创建 ViewModel:

val arguments = CarFragmentArgs.fromBundle(requireArguments())
val application = requireNotNull(this.activity).application
val dataSource = AppDatabase.getInstance(application).refuelingDAO
val viewModelFactory = CarFragmentVMFactory(dataSource, arguments.carID, application)

viewModel = ViewModelProvider(this, viewModelFactory).get(CarFragmentVM::class.java)

binding.carViewModel = viewModel
binding.lifecycleOwner = this
binding.buttonTest.setOnClickListener {
    viewModel.listAll()
}

视图模型:

class CarFragmentVM(
    private val database: RefuelingDAO,
    private val carID: Long,
    application: Application
) :
        AndroidViewModel(application)
{
    val refueling = database.getAll()
   
    fun listAll()
    {
        refueling.value!!.forEach { \\ IN THIS PLACE I GET ERROR
            Log.d(it)
        }
    }
}

当我点击“buttonTest”时,我得到了KotlinNullPointerException。正如我所说,我在另一个片段中有相同的代码(我只是找不到任何差异),它工作正常,但事实并非如此。谁能看到我的问题出在哪里或提供任何线索我应该在哪里寻找它?

Logcat:

E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.example.fuelmanager, PID: 22828
    kotlin.KotlinNullPointerException
        at com.example.fuelmanager.ui.car.CarFragmentVM.listAll(CarFragmentVM.kt:44)
        at com.example.fuelmanager.ui.car.CarFragment$onCreateView$1.onClick(CarFragment.kt:44)
        at android.view.View.performClick(View.java:7125)
        at android.view.View.performClickInternal(View.java:7102)
        at android.view.View.access$3500(View.java:801)
        at android.view.View$PerformClick.run(View.java:27336)
        at android.os.Handler.handleCallback(Handler.java:883)
        at android.os.Handler.dispatchMessage(Handler.java:100)
        at android.os.Looper.loop(Looper.java:214)
        at android.app.ActivityThread.main(ActivityThread.java:7356)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:492)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:930)

第 44 行是

refueling.value!!.forEach { \\ IN THIS PLACE I GET ERROR
            Log.d(it)
        }

【问题讨论】:

  • "当我点击按钮时,我得到了 KotlinNullPointerException。" -- 使用 Logcat 检查堆栈跟踪。如果您不了解堆栈跟踪,请编辑您的问题并提供堆栈跟踪。
  • @CommonsWare 好的,我会添加,但在我的代码中只有一个按钮,我指出了一个我得到错误的代码
  • 是的,但是代码中有很多级别可以触发NullPointerException。就目前而言,你让我们猜测。 始终提供堆栈跟踪,无论是堆栈溢出问题、GitHub 存储库中的错误报告、JIRA 中的错误报告,还是您报告崩溃的其他任何地方。
  • @CommonsWare 是的,你是对的。我加了

标签: android sqlite kotlin android-room dao


【解决方案1】:

getAll() 正在返回 LiveData。实际的数据库查询是在后台线程上完成的。这需要一些时间才能完成。同时,valuenull。但是,您假设它将立即完成。因此,您尝试立即使用value,然后您会因NullPointerException 而崩溃。

observe()LiveData

【讨论】:

  • 谢谢,它正在工作,但我的代码中缺少部分内容。但我就是不明白为什么添加空观察者`viewModel.refueling.observe(viewLifecycleOwner, Observer {})` 会使这段代码工作。当我单击按钮时,我试图立即获取值,就像以前一样。 Observer 仅让我能够在每次 LivaData 更改时执行一段代码,但当我单击按钮时它不会执行。
  • @iknow:“但我就是不明白为什么添加空观察者 `viewModel.refueling.observe(viewLifecycleOwner, Observer {})` 会使这段代码工作”——那是因为它不是工作,至少不可靠。将使用LiveData-emitted 结果的逻辑放在Observer 中。
  • 是的,是的,我知道并且我想在创建 RecyclerView 之后执行它,就像在上一个片段中一样(这就是为什么我没有在这个片段中添加观察者,因为我没有 RycaclerView 并且我只是想测试数据库)。但是此时添加空观察者viewModel.refueling.observe(viewLifecycleOwner, Observer {}) 使我的代码可以正常工作,这有点奇怪
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-01-01
  • 2021-09-17
  • 1970-01-01
  • 1970-01-01
  • 2019-07-01
  • 1970-01-01
相关资源
最近更新 更多