Не удалось создать конвертер для моего класса в Android Retrofit library

Я мигрирую от использования Volley to Retrofit, у меня уже есть класс gson, который я использовал ранее для преобразования ответа JSONObject в объект, реализующий аннотации gson. Когда я пытаюсь заставить http получить запрос с помощью модификации, но потом мое приложение вылетает с этой ошибкой:

 Unable to start activity ComponentInfo{com.lightbulb.pawesome/com.example.sample.retrofit.SampleActivity}: java.lang.IllegalArgumentException: Unable to create converter for class com.lightbulb.pawesome.model.Pet
    for method GitHubService.getResponse

Im, следуя руководству в retrofit, и я придумываю эти реализации:

Это моя деятельность, когда я пытаюсь выполнить ретро-запрос http:

public class SampleActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_sample);

        Retrofit retrofit = new Retrofit.Builder()
                .baseUrl("**sample base url here**")
                .build();

        GitHubService service = retrofit.create(GitHubService.class);
        Call<Pet> callPet = service.getResponse("41", "40");
        callPet.enqueue(new Callback<Pet>() {
            @Override
            public void onResponse(Response<Pet> response) {
                Log.i("Response", response.toString());
            }

            @Override
            public void onFailure(Throwable t) {
                Log.i("Failure", t.toString());
            }
        });
        try{
            callPet.execute();
        } catch (IOException e){
            e.printStackTrace();
        }

    }
}

Мой интерфейс, который оказался моим API

public interface GitHubService {
    @GET("/ **sample here** /{petId}/{otherPet}")
    Call<Pet> getResponse(@Path("petId") String userId, @Path("otherPet") String otherPet);
}

И, наконец, класс Pet, который должен быть ответом:

public class Pet implements Parcelable {

    public static final String ACTIVE = "1";
    public static final String NOT_ACTIVE = "0";

    @SerializedName("is_active")
    @Expose
    private String isActive;
    @SerializedName("pet_id")
    @Expose
    private String petId;
    @Expose
    private String name;
    @Expose
    private String gender;
    @Expose
    private String age;
    @Expose
    private String breed;
    @SerializedName("profile_picture")
    @Expose
    private String profilePicture;
    @SerializedName("confirmation_status")
    @Expose
    private String confirmationStatus;

    /**
     *
     * @return
     * The confirmationStatus
     */
    public String getConfirmationStatus() {
        return confirmationStatus;
    }

    /**
     *
     * @param confirmationStatus
     * The confirmation_status
     */
    public void setConfirmationStatus(String confirmationStatus) {
        this.confirmationStatus = confirmationStatus;
    }

    /**
     *
     * @return
     * The isActive
     */
    public String getIsActive() {
        return isActive;
    }

    /**
     *
     * @param isActive
     * The is_active
     */
    public void setIsActive(String isActive) {
        this.isActive = isActive;
    }

    /**
     *
     * @return
     * The petId
     */
    public String getPetId() {
        return petId;
    }

    /**
     *
     * @param petId
     * The pet_id
     */
    public void setPetId(String petId) {
        this.petId = petId;
    }

    /**
     *
     * @return
     * The name
     */
    public String getName() {
        return name;
    }

    /**
     *
     * @param name
     * The name
     */
    public void setName(String name) {
        this.name = name;
    }

    /**
     *
     * @return
     * The gender
     */
    public String getGender() {
        return gender;
    }

    /**
     *
     * @param gender
     * The gender
     */
    public void setGender(String gender) {
        this.gender = gender;
    }

    /**
     *
     * @return
     * The age
     */
    public String getAge() {
        return age;
    }

    /**
     *
     * @param age
     * The age
     */
    public void setAge(String age) {
        this.age = age;
    }

    /**
     *
     * @return
     * The breed
     */
    public String getBreed() {
        return breed;
    }

    /**
     *
     * @param breed
     * The breed
     */
    public void setBreed(String breed) {
        this.breed = breed;
    }

    /**
     *
     * @return
     * The profilePicture
     */
    public String getProfilePicture() {
        return profilePicture;
    }

    /**
     *
     * @param profilePicture
     * The profile_picture
     */
    public void setProfilePicture(String profilePicture) {
        this.profilePicture = profilePicture;
    }


    protected Pet(Parcel in) {
        isActive = in.readString();
        petId = in.readString();
        name = in.readString();
        gender = in.readString();
        age = in.readString();
        breed = in.readString();
        profilePicture = in.readString();
    }

    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(isActive);
        dest.writeString(petId);
        dest.writeString(name);
        dest.writeString(gender);
        dest.writeString(age);
        dest.writeString(breed);
        dest.writeString(profilePicture);
    }

    @SuppressWarnings("unused")
    public static final Parcelable.Creator<Pet> CREATOR = new Parcelable.Creator<Pet>() {
        @Override
        public Pet createFromParcel(Parcel in) {
            return new Pet(in);
        }

        @Override
        public Pet[] newArray(int size) {
            return new Pet[size];
        }
    };
}

Ответ 1

До 2.0.0 конвертер по умолчанию был gson-конвертером, но в 2.0.0, а позже конвертер по умолчанию - ResponseBody. Из документов:

По умолчанию Retrofit может только десериализовать тела HTTP в OkHttp ResponseBody, и он может принимать только его тип RequestBody для @Body.

В 2.0.0+ вам нужно явно указать, что вам нужен конвертер Gson:

Retrofit retrofit = new Retrofit.Builder()
    .baseUrl("**sample base url here**")
    .addConverterFactory(GsonConverterFactory.create())
    .build();

Вам также потребуется добавить следующую зависимость в ваш файл gradle:

compile 'com.squareup.retrofit2:converter-gson:2.1.0'

Используйте ту же версию для конвертера, что и для вашей модификации. Вышеуказанная соответствует этой зависимой модификации:

compile ('com.squareup.retrofit2:retrofit:2.1.0')

Кроме того, обратите внимание, что при записи этого документа модифицированные документы не полностью обновлены, поэтому этот пример вызвал у вас проблемы. Из документов:

Примечание. Этот сайт все еще находится в процессе расширения для новых API 2.0.

Ответ 2

Если кто-нибудь когда-нибудь столкнется с этим в будущем, потому что вы пытаетесь определить свою собственную фабрику пользовательских преобразователей и получаете эту ошибку, это также может быть вызвано наличием нескольких переменных в классе с орфографической ошибкой или одним и тем же сериализованным именем. IE:

public class foo {
  @SerializedName("name")
  String firstName;
  @SerializedName("name")
  String lastName;
}

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

Обновление: имейте в виду, что эта логика также сохраняется посредством наследования. Если вы расширите родительский класс с помощью объекта с таким же сериализованным именем, как и в подклассе, это вызовет ту же проблему.

Ответ 3

Основываясь на верхнем комментарии, я обновил свой импорт

implementation 'com.squareup.retrofit2:retrofit:2.1.0'
implementation 'com.squareup.retrofit2:converter-gson:2.1.0'

Я использовал http://www.jsonschema2pojo.org/ для того, чтобы создать pojo из результатов Spotify json и убедиться, что указан формат Gson.

В эти дни есть плагины Android Studio, которые могут создавать модели данных pojo или Kotlin для вас. Отличный вариант для Mac - Quicktype. https://itunes.apple.com/us/app/paste-json-as-code-quicktype/id1330801220

Ответ 4

В моем случае у меня был объект TextView внутри моего модального класса, и GSON не знал, как его сериализовать. Пометив его как "переходный", проблема решена.

Ответ 5

Пост @Silmarilos помог мне решить эту проблему. В моем случае это было то, что я использовал "id" как сериализованное имя, например так:

 @SerializedName("id")
var node_id: String? = null

и я изменил это на

 @SerializedName("node_id")
var node_id: String? = null

Все работает сейчас. Я забыл, что "id" является атрибутом по умолчанию.

Ответ 6

Это может помочь кому-то

В моем случае по ошибке я написал SerializedName, как это

@SerializedName("name","time")
String name,time; 

Так должно быть

@SerializedName("name")
String name;

@SerializedName("time")
String time;

Ответ 7

Эй, сегодня я сталкивался с той же проблемой, и мне потребовался целый день, чтобы найти решение, но это решение, которое я наконец нашел. Я использую Dagger в своем коде, и мне нужно было реализовать конвертер Gson в моем экземпляре модернизации.

так что это был мой код раньше

@Provides
    @Singleton
    Retrofit providesRetrofit(Application application,OkHttpClient client) {
        String SERVER_URL=URL;
        Retrofit.Builder builder = new Retrofit.Builder();
        builder.baseUrl(SERVER_URL);
        return builder
                .client(client)
                .build();
    }

это то, что я закончил

@Provides
    @Singleton
    Retrofit providesRetrofit(Application application,OkHttpClient client, Gson gson) {
        String SERVER_URL=URL;
        Retrofit.Builder builder = new Retrofit.Builder();
        builder.baseUrl(SERVER_URL);
        return builder
                .client(client)
                .addConverterFactory(GsonConverterFactory.create(gson))
                .build();
    }

обратите внимание на то, что в первом примере нет конвертера, и если вы не создали Gson, добавьте его так:

    @Provides
    @Singleton
    Gson provideGson() {
        GsonBuilder gsonBuilder = new GsonBuilder();

   gsonBuilder.setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES);
        return gsonBuilder.create();
    }

и убедитесь, что вы включили его в вызов метода для модернизации.

еще раз надеюсь, что это поможет кому-то, как я.

Ответ 8

В моем случае это было связано с попыткой передать список, возвращаемый моим сервисом, в ArrayList. Итак, что у меня было:

@Json(name = "items")
private ArrayList<ItemModel> items;

когда я должен был

@Json(name = "items")
private List<ItemModel> items;

Надеюсь, это кому-нибудь поможет!

Ответ 9

В моем случае проблема заключалась в том, что в моей модели SUPERCLASS было определено это поле. Очень глупо, я знаю....