Retrofit2 Android: ожидалось BEGIN_ARRAY, но было BEGIN_OBJECT в строке 1, столбце 2, путь $

Я знаю, что это не первый раз, когда кто-то спрашивает об этой проблеме, но с Retrofit2 я не могу найти правильного решения моей проблемы. Я последовал онлайн-руководству, и он работал нормально. Когда я применил тот же код к своей конечной точке, я получил следующее исключение: java.lang.IllegalStateException: Expected BEGIN_ARRAY but was BEGIN_OBJECT at line 1 column 2 path $ Я не знаю, как решить эту проблему.

Интерфейс:

public interface MyApiService {

// Is this right place to add these headers?
@Headers({"application-id: MY-APPLICATION-ID",
        "secret-key: MY-SECRET-KEY",
        "application-type: REST"})
@GET("Music")
Call<List<Music>> getMusicList();



Retrofit retrofit = new Retrofit.Builder()
        .baseUrl(MySettings.REST_END_POINT)
        .addConverterFactory(GsonConverterFactory.create())
        .build();
}

Код клиента:

MyApiService service = MyApiService.retrofit.create(MyApiService.class);
Call<List<Music>> call = service.getMusicList();
call.enqueue(new Callback<List<Music>>() {

    @Override
    public void onResponse(Call<List<Music>> call, Response<List<Music>> response) {
        Log.e("MainActivity", response.body().
    }

    @Override
    public void onFailure(Call<List<Music>> call, Throwable t) {
        Log.e("MainActivity", t.toString());
    }
});

Этот код работает с этой полезной нагрузкой:

[
{
    "login": "JakeWharton",
    "id": 66577,
    "avatar_url": "https://avatars.githubusercontent.com/u/66577?v=3",
    "gravatar_id": "",
    "url": "https://api.github.com/users/JakeWharton",
    "html_url": "https://github.com/JakeWharton",
    "followers_url": "https://api.github.com/users/JakeWharton/followers",
    "following_url": "https://api.github.com/users/JakeWharton/following{/other_user}",
    "gists_url": "https://api.github.com/users/JakeWharton/gists{/gist_id}",
    "starred_url": "https://api.github.com/users/JakeWharton/starred{/owner}{/repo}",
    "subscriptions_url": "https://api.github.com/users/JakeWharton/subscriptions",
    "organizations_url": "https://api.github.com/users/JakeWharton/orgs",
    "repos_url": "https://api.github.com/users/JakeWharton/repos",
    "events_url": "https://api.github.com/users/JakeWharton/events{/privacy}",
    "received_events_url": "https://api.github.com/users/JakeWharton/received_events",
    "type": "User",
    "site_admin": false,
    "contributions": 741
},
{....

но не с этим:

{
"offset": 0,
"data": [
    {
        "filename": "E743_1458662837071.mp3",
        "created": 1458662854000,
        "publicUrl": "https://api.backendless.com/dbb77803-1ab8-b994-ffd8-65470fa62b00/v1/files/music/E743_1458662837071.mp3",
        "___class": "Music",
        "description": "",
        "likeCount": 0,
        "title": "hej Susanne. ",
        "ownerId": "E743756F-E114-6892-FFE9-BCC8C072E800",
        "updated": null,
        "objectId": "DDD8CB3D-ED66-0D6F-FFA5-B14543ABC800",
        "__meta": "{\"relationRemovalIds\":{},\"selectedProperties\":[\"filename\",\"created\",\"publicUrl\",\"___class\",\"description\",\"likeCount\",\"title\",\"ownerId\",\"updated\",\"objectId\"],\"relatedObjects\":{}}"
    },
    {...

Мой музыкальный класс:

public class Music {

   private String ownerId;
   private String filename;
   private String title;
   private String description;
   private String publicUrl;
   private int likeCount;

   // Getters & Setters

}

person oxyt    schedule 23.03.2016    source источник
comment
посмотрите здесь . Вам нужно вернуть другой класс, содержащий List<Music> data   -  person Blackbelt    schedule 23.03.2016
comment
Он ожидает массив, вы даете ему объект   -  person Tim    schedule 23.03.2016


Ответы (2)


Когда вы говорите: «Этот код работает с этой полезной нагрузкой: ... но не с этим: ...», это ожидается, и вот как это должно работать. Фактически, сообщение об ошибке сообщает вам, что при преобразовании json в объект java вызов ожидал массив в json, но вместо этого получил объект.

Этот звонок:

@GET("Music")
Call<List<Music>> getMusicList();

ожидает список из Music объектов, поэтому работает с json:

[
  {
    "login": "JakeWharton",
    ...
  },
  ...
]

Поскольку сам json является массивом ваших Music объектов (Retrofit может преобразовывать массивы json в списки java). Для второго json у вас есть просто объект, а не массив (обратите внимание на отсутствие [...]). Для этого вам нужно создать еще один вызов с другой моделью, которая сопоставляется с этим json. Предположим, вы назвали модель MusicList. Вот как может выглядеть звонок:

@GET("Music")
Call<MusicList> getMusicList();

(Обратите внимание, что вам может потребоваться изменить имя метода, если вы хотите сохранить и первый вызов, и этот).

Модель MusicList может выглядеть примерно так:

public class MusicList {
  @SerializedName("data")
  private List<Music> musics;
  // ...
}

Я предполагаю, что массив data представляет собой список объектов Music, но я заметил, что jsons совершенно разные. Возможно, вам придется адаптировать это, но я думаю, что вы уловили картину.

person Fred    schedule 23.03.2016
comment
Большое спасибо за то, что поделились этим, я весь день пытался найти решение, и благодаря вашим советам оно сработало. Большое тебе спасибо!! - person RafaelB13; 22.05.2020

Встречаюсь с этой проблемой. Потому что playload - это объект, а не массив объектов. Поэтому я удаляю список.

Пример кода

UserAPI.java

public interface UserAPI {
    @GET("login/cellphone")
    Call<LoginResponse> login(@Query("phone") String phone, 
                              @Query("password") String password);
}

Код звонка

Retrofit retrofit = new Retrofit
        .Builder()
        .addConverterFactory(GsonConverterFactory.create())
        .baseUrl(Constant.CLOUD_MUSIC_API_BASE_URL)
        .build();

UserAPI userAPI = retrofit.create(UserAPI.class);

userAPI.login(phone, password).enqueue(new Callback<LoginResponse>() {
    @Override
    public void onResponse(Call<LoginResponse> call, Response<LoginResponse> response) {
        System.out.println("onResponse");
        System.out.println(response.body().toString());
    }

    @Override
    public void onFailure(Call<LoginResponse> call, Throwable t) {
        System.out.println("onFailure");
        System.out.println(t.fillInStackTrace());
    }
});
person shellhub    schedule 13.09.2018
comment
Примечание: использование GET в качестве метода входа в систему имеет некоторые проблемы с безопасностью, вместо этого вам следует выбрать POST. Пожалуйста, проверьте это stackoverflow.com/a/5868860/3503855 - person Anis LOUNIS; 03.09.2019