没有 LiveData 的房间

Shi*_*vam 14 android android-room android-livedata

我正在试验Room数据库。我不希望我的数据被观察到,我只想从数据库中获取一次数据。如何使用 MVVM 实现这一目标?

我面临的问题:如果我尝试在没有数据的情况下获取数据AsyncTask:无法访问主线程上的数据库,因为它可能会长时间锁定 UI(如预期),如果我使用AsyncTask,该方法返回 nullList作为方法在AsyncTask完成之前返回。

道类:

@Query("SELECT * FROM student_table where StudentName = :studentName")List<Student> getStudentWithSameName(String studentName);
Run Code Online (Sandbox Code Playgroud)

存储库:

public List<Student> getAllStudentWithSameName(String studentName) {
    new GetAllStudentWithSameNameAsyncTask(studentDao).execute(studentName);
    return studentsWithSameName;
}



private class GetAllStudentWithSameNameAsyncTask extends AsyncTask< String,Void, List<Student> > {

    StudentDao studentDao;

    public GetAllStudentWithSameNameAsyncTask(StudentDao studentDao) {
        this.studentDao = studentDao;
    }

    @Override
    protected List<Student> doInBackground(String... strings) {
        List<Student> students = studentDao.getStudentWithSameName(strings[0]);
        return students;
    }

    @Override
    protected void onPostExecute(List<Student> students) {
        studentsWithSameName = students;
        super.onPostExecute(students);
    }
}
Run Code Online (Sandbox Code Playgroud)

视图模型:

public List<Student> getStudentWithSameName(String studentName) {
    studentsWithSameName = studentRepository.getAllStudentWithSameName(studentName);
    return studentsWithSameName;
}
Run Code Online (Sandbox Code Playgroud)

主要活动:

viewModel = ViewModelProviders.of(this).get(StudentViewModel.class);
List<Student> students = viewModel.getStudentWithSameName("Bill");
Run Code Online (Sandbox Code Playgroud)

Cul*_*lub 6

您需要使用异步(“挂起”)函数,因为数据库调用可能需要很长时间。然后,要使用结果,您必须在完成时调用代码块,而不是立即运行它。


在我的YourClassDao.kt更改funsuspend fun,并且LiveData<List<YourClass>>只是List<YourClass>

// original: this returns a LiveData object
@Query("SELECT * FROM my_table WHERE my_field = :myId")
fun getMyObject(myId: String): LiveData<List<YourClass>>
Run Code Online (Sandbox Code Playgroud)

变成:

// new: this returns a normal object
@Query("SELECT * FROM my_table WHERE my_field = :myId")
suspend fun getMyObject(myId: String): List<YourClass>
Run Code Online (Sandbox Code Playgroud)

要使用数据,您需要启动异步作业来获取数据,然后启动invokeOnCompletion使用数据所需的代码。

// using the second (suspend fun) version from above
fun useMyData() {
    val database = AppDatabase.getInstance(context).YourClassDao()  // context could be an activity, for example.

    // start an async job to get the data
    val getDataJob = GlobalScope.async { database.getMyObject("someId") }

    // tell the job to invoke this code when it's done
    getDataJob.invokeOnCompletion { cause ->
        if (cause != null) {
            // error!  Handle that here
            Unit
        } else {
            val myData = getDataJob.getCompleted()

            // ITEM 1
            // ***************************
            // do something with your data
            // ***************************

            Unit  // this is just because the lambda here has to return Unit
        }
    }

    // ITEM 2 - this might happen before ITEM 1
} 
Run Code Online (Sandbox Code Playgroud)