Не удалось создать конвертер для моего класса в 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 было определено это поле. Очень глупо, я знаю....