java - 如何在 android 的 ViewModel 中从一个 LiveData 对象获取数据到另一个对象?
问题描述
我正在尝试创建一个通过房间和 LiveData 使用 SQL 数据库的应用程序。这是一个非常非常简单的测验应用程序。将显示一个问题和几个可能的答案。此外,还有一个下一个和上一个按钮来浏览问题。
有两个表:“table_question”和“table_answer”。
table_question 由以下列组成:id, question
table_answer: id, answer, id_question
id_question 分配问题的答案,因为它是 1:n 关系。
我从数据库中获取所有问题和所有答案的列表作为 LiveData 对象。另一方面,我想要一个 LiveData 类 QuizWorld 来处理所有需要在屏幕上显示的数据(回答的问题、正确回答的问题、错误回答的问题、留下的问题等等)。Activity 将观察此类,以便更新 UI。
但是如何缩小两个 LiveData 列表“问题”和“答案”与创建 LiveData QuizWorld 之间的差距?QuizWorld 将需要保存在问题和答案中的那些信息,但 LiveData 对象没有构造函数。
我有使用 Transformations.switchMap 的想法,但我无法让它发挥作用。即使我忘记了 QuizWorld 类并在 ViewModel 中执行所有操作,也会给我留下无法访问 LiveData 列表的问题。
注意:列表实际上没有必要是 LiveData 对象,因为 SQL 数据库在运行时不会更改。我只是不知道如何通过房间访问数据库。但是,如果将来数据库变得更加动态(尚未计划),它仍然是一个不错的功能。
这是代码的一部分:
public class MyViewModel extends AndroidViewModel{
private Repository mRepository;
private LiveData<List<Question>> questions;
private LiveData<List<Answer>> answers;
private LiveData<QuizWorld> mQuizWorld;
public MyViewModel (Application application) {
super(application);
mRepository = new Repository(application);
questions = mRepository.getQuestions;
answers = mRepository.getAnswers;
!!!!!Here should be the missing code I have no idea for!!!!!
}
public LiveData<QuizWorld> getQuizWorld(){
return mQuizWorld;
}
QuizWorld 类:
public class QuizWorld{
private List<Question> questions;
private List<Answer> answers;
private int score;
public QuizWorld(List<Question> questions, List<Answer> answers){
this.questions = questions;
this.answers = answers;
init();
}
private void init() {
//do things with the questions and answers list
}
public int getScore(){
return score;
}
}
主要活动:
public class MainActivity {
private LiveData<QuizWorld> mQuizWorld;
private MyViewModel myViewModel;
private TextView tvScore;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedinstanceState);
setContenView(...);
tvScore = (TextView) findViewById(R.id.tv_Score);
myViewModel = ViewModelProvider.of(this).get(MyViewModel.class)
mQuizViewModel.getQuizWorld().observe(this, new
Observer<QuizWorld>() {
@Override
public void onChanged(QuizWorld quizWorld){
tvScore.setText(quizWorld.getScore));
//show count of questions wrongly answered
//show count of questions remaining
//...
}
}
}
}
解决方案
我不完全确定你到底想要什么,但我宁愿建议你有一个包含一个问题及其答案的对象 QuestionWithAnswer 而不是两个不同的列表。然后,您可以在 dao 类中进行查询,该查询将返回该 QuestionWithAnswer 类(或在您的情况下为这些对象的列表)
请参阅此链接以查询多个表(包含用户和宠物的示例应该看起来像您的)在 Dao 类中,您将有类似的内容:
@Query("SELECT id as isQuestion, question, id as idAnswer, answer " +
"FROM table_question " +
"LEFT JOIN table_answer on table_question.id = table_answer.id_question")
public LiveData<List<QuestionWithAnswer>> loadResults();
如果您想通过一个查询直接在另一个对象中包含对象,另请参阅此链接。
或者,您可以进行转换,但我认为它不会非常有效,您会执行更多 SQL 查询,这通常对性能来说不是超级好(如果您有一个小型数据库,您不会注意到它,但是其他具有更多数据的样本)。
编辑 :
如果需要转换 QuizWorld 对象中的 List,可以使用 Transformations.map 方法,如下所示:
quizWorldLiveData = Transformations.map(questionsWithAnswersLiveData, (data) -> new QuizWorld(data))
推荐阅读
- c# - 使用简短的初始化语法来初始化对值列表
- ios - Vuetify 数据表在 Safari 中呈现问题
- html - 动画延迟问题
- python - 是否可以从 Python 运行 Falcon 应用程序?
- android - 如何为 wrap_content 设置动画?
- java - 尝试将文件上传到 Amazon S3 时返回错误代码 400
- azure - Azure:从 MySQL 数据库创建 O365 事件
- docker - Docker In Docker IN Docker Compose(网络问题)
- r - 如何仅使用一定数量的值逐行删除数据
- ruby-on-rails - 未定义的方法“client_path”