首页 > 解决方案 > 如何在 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
                        //...
                   }
              }
         }
    }

标签: javaandroiddatabaseandroid-architecture-componentsandroid-livedata

解决方案


我不完全确定你到底想要什么,但我宁愿建议你有一个包含一个问题及其答案的对象 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))

推荐阅读