Как сделать вызов API модификации с помощью ViewModel и LiveData

это первый раз, когда я пытаюсь реализовать архитектуру MVVM, и я немного не понимаю, как правильно сделать вызов API.

В настоящее время я просто пытаюсь сделать простой запрос из API IGDB и вывести имя первого элемента в журнале.

Моя деятельность настроена следующим образом:

public class PopularGamesActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_popular_games);


        PopularGamesViewModel popViewModel = ViewModelProviders.of(this).get(PopularGamesViewModel.class);
        popViewModel.getGameList().observe(this, new Observer<List<Game>>() {
            @Override
            public void onChanged(@Nullable List<Game> gameList) {
                String firstName = gameList.get(0).getName();
                Timber.d(firstName);
            }
        });
    }
}

Моя модель просмотра настроена следующим образом:

public class PopularGamesViewModel extends AndroidViewModel {

    private static final String igdbBaseUrl = "https://api-endpoint.igdb.com/";
    private static final String FIELDS = "id,name,genres,cover,popularity";
    private static final String ORDER = "popularity:desc";
    private static final int LIMIT = 30;

    private LiveData<List<Game>> mGameList;

    public PopularGamesViewModel(@NonNull Application application) {
        super(application);


        // Create the retrofit builder
        Retrofit.Builder builder = new Retrofit.Builder()
                .baseUrl(igdbBaseUrl)
                .addConverterFactory(GsonConverterFactory.create());

        // Build retrofit
        Retrofit retrofit = builder.build();

        // Create the retrofit client
        RetrofitClient client = retrofit.create(RetrofitClient.class);
        Call<LiveData<List<Game>>> call = client.getGame(FIELDS,
                ORDER,
                LIMIT);

        call.enqueue(new Callback<LiveData<List<Game>>>() {
            @Override
            public void onResponse(Call<LiveData<List<Game>>> call, Response<LiveData<List<Game>>> response) {
                if (response.body() != null) {
                    Timber.d("Call response body not null");
                    mGameList = response.body();

                } else {
                    Timber.d("Call response body is null");
                }
            }

            @Override
            public void onFailure(Call<LiveData<List<Game>>> call, Throwable t) {
                Timber.d("Retrofit call failed");
            }
        });

    }

    public LiveData<List<Game>> getGameList() {
        return mGameList;
    }

Теперь проблема в том, что это вызов API, начальное значение mGameList будет нулевым, пока call.enqueue не вернется со значением. Это вызовет исключение нулевого указателя с

popViewModel.getGameList().observe(this, new Observer<List<Game>>() {
  1. Итак, как правильно вести наблюдение за LiveData при выполнении вызова API?
  2. Правильно ли я выполнил вызов Retrofit API?

person Chao Li    schedule 10.08.2018    source источник


Ответы (2)


В вашем коде 3 проблемы.

  1. Вы должны создать объект MutableLiveData, потому что у вас есть пустой ответ перед вызовом API, тогда ваш объект LiveData будет каким-то образом заполнен через ответ IGDB.
private MutableLiveData<List<Game>> mGameList = new MutableLiveData();
//...
public LiveData<List<Game>> getGameList() {
    return mGameList;
}
  1. Другая ошибка заключается в изменении ссылки на mGameList вместо установки его значения, поэтому попробуйте изменить:
Timber.d("Call response body not null");
mGameList = response.body();

to

mGameList.setValue(response.body());
  1. Вызов модернизации внутри вашего ViewModel класса позволяет избежать разделения задач. Лучше создать модуль репозитория и получать ответ через интерфейс. Подробнее читайте в этой статье.

Модули репозитория отвечают за обработку операций с данными. Они предоставляют чистый API для остальной части приложения. Они знают, откуда брать данные и какие вызовы API выполнять при обновлении данных. Вы можете рассматривать их как посредников между различными источниками данных (постоянная модель, веб-сервис, кеш и т. Д.).

person Saeed Masoumi    schedule 10.08.2018
comment
Благодарю за ваш ответ. Я действительно заметил свои ошибки 1 и 2, прежде чем понял ваш ответ, поэтому я исправил это. Спасибо, что указали на мою ошибку 3, я пойду добавлю репозиторий. - person Chao Li; 10.08.2018
comment
Думаю, у многих людей такая же проблема, спасибо, ты спас мне жизнь. - person Richard Hsieh; 24.02.2019
comment
Как я могу узнать из View, когда запрос данных репозитория завершен. - person Alston; 12.10.2019

Я просто вдохновился демонстрацией Google и создал библиотеку, которая может добавить поддержку LiveData для Retrofit. Использование простое:

Retrofit.Builder()
            .baseUrl("https://api.github.com/")
            .addConverterFactory(GsonConverterFactory.create())
            .addCallAdapterFactory(LiveDataCallAdapterFactory())
            .build()
            .create(GithubService::class.java)
            .getUser("shawnlinboy").observe(this,
                Observer { response ->
                    when (response) {
                        is ApiSuccessResponse -> {
                          //success response
                        }
                        else -> {
                            //failed response
                        }
                    }
                })

Сайт библиотеки: https://github.com/shawnlinboy/retrofit-livedata-adapter

person shawnlinboy    schedule 28.08.2019