如何使用 Volley for Android 存储来自 REST API 的数据

How to store data from REST API with Volley for Android

我尝试从我的 Api 获取数据,我可以获取数据并将其转换为对象 class 我创建并显示了它,但我无法将所有对象实例存储在列表中.有人知道吗? 我尝试添加我在列表中获得的每个元素,然后尝试从中获取元素但不起作用,列表保持为空。 这就是我的 API returns :

{"_id":"60683c5bcdfcb74689bc8382","questionId":3,"question":"Question 3","choices":"1-/-2-/-3-/-4","category":"Other","difficulty":"Easy"}

这是我的“Api测试activity”:

package com.jojo.jojozquizz;

import android.os.Bundle;
import android.util.Log;
import android.widget.TextView;

import androidx.appcompat.app.AppCompatActivity;

import com.android.volley.Request;
import com.android.volley.RequestQueue;
import com.android.volley.Response;
import com.android.volley.VolleyError;
import com.android.volley.toolbox.JsonObjectRequest;
import com.android.volley.toolbox.Volley;
import com.jojo.jojozquizz.model.Question;
import com.jojo.jojozquizz.model.QuestionBank;
import com.jojo.jojozquizz.tools.APIListener;

import org.json.JSONException;
import org.json.JSONObject;

public class ApiTests extends AppCompatActivity implements APIListener {

    private TextView text;

    public static final String TAG = "MyTag";

    private QuestionBank mQuestionBank;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_api_tests);
        mQuestionBank = new QuestionBank();
        text = findViewById(R.id.api_text);

        String url = "https://nextfor.studio/questions/test";

        getQuestionFromApi(url);
    }

    private void getQuestionFromApi(String url) {
        APIListener listener = this;
        RequestQueue requestQueue = Volley.newRequestQueue(this);
        JsonObjectRequest jsonObjectRequest = new JsonObjectRequest(Request.Method.GET, url, null, new Response.Listener<JSONObject>() {
            @Override
            public void onResponse(JSONObject response) {
                try {
                    Log.d(TAG, "onResponse: " + response.toString());
                    Question question = new Question();
                    question.setId(response.getInt("questionId"));
                    question.setQuestion(response.getString("question"));
                    question.setChoices(response.getString("choices"));
                    question.setCategory(response.getString("category"));
                    question.setDifficulty(response.getString("difficulty"));

                    listener.questionReceived(question);
                } catch (JSONException e) {
                    Log.i(TAG, e.getMessage());
                }
            }
        }, new Response.ErrorListener() {
            @Override
            public void onErrorResponse(VolleyError error) {

                Log.d(TAG, "onErrorResponse: " + error.getMessage());
            }
        });
        requestQueue.add(jsonObjectRequest);
    }

    @Override
    public void questionReceived(Question q) {
        mQuestionBank.addQuestion(q);
    }
}

问题class:

package com.jojo.jojozquizz.model;

import androidx.room.ColumnInfo;
import androidx.room.Entity;
import androidx.room.Ignore;
import androidx.room.PrimaryKey;

import java.io.Serializable;
import java.util.List;

@Entity(tableName = "questions")
public class Question implements Serializable {

    @PrimaryKey()
    @ColumnInfo(name = "id")
    private long id;

    @ColumnInfo(name = "question")
    private String mQuestion;

    @Ignore
    private List<String> mChoiceList;

    @ColumnInfo(name = "choices")
    private String mChoices;

    @ColumnInfo(name = "answer_index")
    private int mAnswerIndex;

    @ColumnInfo(name = "categorie")
    private String mCategory;

    @Ignore
    private String mTrueAnswer;

    @ColumnInfo(name = "difficulty")
    private String mDifficulty;

    public Question() {
    }

    public Question(int id, String mQuestion, List<String> mChoiceList, String mCategorie, String mDifficulty) {
        this.id = id;
        this.mQuestion = mQuestion;
        this.mChoices = mChoiceList.get(0) + "-/-" + mChoiceList.get(1) + "-/-" + mChoiceList.get(2) + "-/-" + mChoiceList.get(3);
        this.mAnswerIndex = 0;
        this.mCategory = mCategorie;
        this.mDifficulty = mDifficulty;
    }

    public Question(String mQuestion, List<String> mChoiceList, String mCategorie, String mDifficulty) {
        this.mQuestion = mQuestion;
        this.mChoices = mChoiceList.get(0) + "-/-" + mChoiceList.get(1) + "-/-" + mChoiceList.get(2) + "-/-" + mChoiceList.get(3);
        this.mAnswerIndex = 0;
        this.mCategory = mCategorie;
        this.mDifficulty = mDifficulty;
    }

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public String getQuestion() {
        return mQuestion;
    }

    public void setQuestion(String question) {
        mQuestion = question;
    }

    public List<String> getChoiceList() {
        return mChoiceList;
    }

    public void setChoiceList(List<String> choiceList) {
        mChoiceList = choiceList;
    }

    public String getChoices() {
        return mChoices;
    }

    public void setChoices(String choices) {
        mChoices = choices;
    }

    public void setAnswerIndex(int mAnswerIndex) {
        this.mAnswerIndex = mAnswerIndex;
    }

    public int getAnswerIndex() {
        return mAnswerIndex;
    }

    public String getCategory() {
        return mCategory;
    }

    public void setCategory(String categorie) {
        mCategory = categorie;
    }

    public String getTrueAnswer() {
        return mTrueAnswer;
    }

    public void setTrueAnswer(String trueAnswer) {
        mTrueAnswer = trueAnswer;
    }

    public String getDifficulty() {
        return mDifficulty;
    }

    public void setDifficulty(String difficulty) {
        mDifficulty = difficulty;
    }
}

QuestionBank(这就像一个问题列表):

package com.jojo.jojozquizz.model;

import java.util.Collections;
import java.util.LinkedList;
import java.util.List;

public class QuestionBank {
  private List<Question> mQuestionList;
  public int mNextQuestionIndex;

  public QuestionBank() {
    mQuestionList = new LinkedList<>();
    mNextQuestionIndex = 0;
  }

  public QuestionBank(List<Question> questionList, boolean shuffle) {
    mQuestionList = questionList;
    if (shuffle)
      Collections.shuffle(mQuestionList);
    mNextQuestionIndex = 0;
  }

  public Question getQuestion() {
    if (mNextQuestionIndex == mQuestionList.size())
      mNextQuestionIndex = 0;

    return mQuestionList.get(mNextQuestionIndex++);
  }

  public Question getQuestion(int i) {
    return  mQuestionList.get(i);
  }

  public void reShuffle() {
    Collections.shuffle(mQuestionList);
  }

  public void addQuestion(Question question) {
    mQuestionList.add(question);
  }

  public int returnListSize() {
    return mQuestionList.size();
  }
}

和API听众:

package com.jojo.jojozquizz.tools;

import com.jojo.jojozquizz.model.Question;

public interface APIListener {
    void questionReceived(Question q);
}

API调用是一个异步过程。所以 getQuestionFromApi(url); 可能需要一些时间,所以如果你在 getQuestionFromApi(url); 行之后尝试 text.setText(mQuestionBank.getQuestion(0).getQuestion());,它不会起作用。您可以使用 LiveData 来处理这种情况。喜欢:

class QuestionBank {
    MutableLiveData<List<Question>> mQuestionList;
    public int mNextQuestionIndex;

    public QuestionBank() {
        mQuestionList = new MutableLiveData<>(new LinkedList<>());
        mNextQuestionIndex = 0;
    }


    public QuestionBank(List<Question> questionList, boolean shuffle) {
        mQuestionList.setValue(questionList);
        if (shuffle)
            Collections.shuffle(mQuestionList.getValue());
        mNextQuestionIndex = 0;
    }

    public Question getQuestion() {
        if (mNextQuestionIndex == mQuestionList.getValue().size())
            mNextQuestionIndex = 0;

        return mQuestionList.getValue().get(mNextQuestionIndex++);
    }

    public Question getQuestion(int i) {
        return  mQuestionList.getValue().get(i);
    }

    public void reShuffle() {
        Collections.shuffle(mQuestionList.getValue());
    }

    public void addQuestion(Question question) {
        List<Question> tempQ =  mQuestionList.getValue();
        tempQ.add(question);
        mQuestionList.setValue(tempQ);
    }

    public int returnListSize() {
        return mQuestionList.getValue().size();
    }
}

并在onCreate()内观察是否添加了任何数据,然后setText()TextView。喜欢:

       QuestionBank questionBank = new QuestionBank();
     questionBank.mQuestionList.observe(getViewLifecycleOwner(), new Observer<List<Question>>() {
          @Override
          public void onChanged(List<Question> questions) {
          if (!questions.isEmpty()){
               textView.setText(questions.get(0).getQuestion());
                Log.d("TAG==>>","New Question added  Total size = "+questions.size());
                }
            }
      });