Пользовательский конвертер для Retrofit 2

Мне нужно обрабатывать динамические ответы JSON.

Раньше я использовал классы и аннотации следующим образом:

public class ChatResponse {

    @SerializedName("status")
    private int status;

    @SerializedName("error")
    private String error;

    @SerializedName("response")
    private Talk response;

    public int getStatus() {
        return status;
    }

    public String getError() {
        return error;
    }

    public Talk getResponse() {
        return response;
    }
}

Когда статус равен 1 (успех), onResponse запускается, и я могу получить объект ChatResponse. Но когда статус равен 0, ответ является ложным в представлении JSON, и он терпит неудачу (onFailure запускается).

Я хочу создать свой собственный конвертер, и у этот вопрос имеет хороший пример, но этот пример относится к модернизации 1.

Мне нужно создать класс, расширяющий Converter.Factory, но я не знаю, как переопределить методы этот класс.

Собственно у меня следующее:

@Override
public Converter<ResponseBody, ?> fromResponseBody(Type type, Annotation[] annotations) {

    return super.fromResponseBody(type, annotations);
}

@Override
public Converter<?, RequestBody> toRequestBody(Type type, Annotation[] annotations) {

    return super.toRequestBody(type, annotations);
}

Как я могу проанализировать ответ JSON самостоятельно на этом этапе?

Заранее спасибо.


person JCarlosR    schedule 19.02.2016    source источник
comment
проверьте этот пример, используя пользовательские десериализаторы   -  person Navneet Krishna    schedule 16.05.2018


Ответы (4)


Я искал простой пример реализации пользовательского конвертера для Retrofit 2. К сожалению, не нашел.

Я нашел это example но, по крайней мере для меня, это слишком сложно для моей цели.

К счастью, я нашел решение.

Это решение заключается в использовании GSON deserializers.

Нам не нужно создавать собственный конвертер, нам просто нужно настроить файл GSON converter.

Вот отличный учебник. А вот код, который я использовал для разбора JSON, описанного в моем вопросе:

person JCarlosR    schedule 20.02.2016
comment
Это добавит фабрику статических преобразователей для того же типа ответа. Что делать, если я хочу сделать синтаксический анализ различных объектов ответа необязательным? У меня могут быть разные ответы для разных API с вариантами и фиксированными объектами ответа. - person Rushi M Thakker; 23.08.2017
comment
Для простого примера вы можете взглянуть на конвертеры скаляров модернизации: github.com/square/retrofit/tree/master/retrofit-converters/ - person simne7; 06.03.2020
comment
Клиент попросил обновить репозиторий как частный. Но я только что обновил ссылки, используя суть GitHub. - person JCarlosR; 01.06.2020

Я нашел решение @JCarlos точным, быстрым и правильным. Мне нужно было реализовать пользовательский конвертер дат для Retrofit 2 на Android. Кажется, вам нужно зарегистрировать сериализатор нового типа в GsonConverterFactory. Реализация выполняется на языке Kotlin.

class RetrofitDateSerializer : JsonSerializer<Date> {
    override fun serialize(srcDate: Date?, typeOfSrc: Type?, context: JsonSerializationContext?): JsonElement? {
        if (srcDate == null)
            return null
        val dateFormat = SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss")
        val formatted = dateFormat.format(srcDate)
        return JsonPrimitive(formatted)
    }
}

и регистрация:

private fun buildGsonConverterFactory(): GsonConverterFactory {
    val gsonBuilder = GsonBuilder()
    // Custom DATE Converter for Retrofit
    gsonBuilder.registerTypeAdapter(Date::class.java, RetrofitDateSerializer())
    return GsonConverterFactory.create(gsonBuilder.create())
}

@Provides @Singleton
internal fun providesRetrofit(applicationContext: Context): Retrofit {
    return Retrofit.Builder()
            .baseUrl(GluApp.Static.BASE_REST_URL_ADDR)
            .addConverterFactory(
                    buildGsonConverterFactory())
            .build()
}
person kosiara - Bartosz Kosarzycki    schedule 16.01.2018

скомпилировать эти две библиотеки для retrofit2

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


import com.lendingkart.prakhar.lendingkartdemo.retrofitPOJOResponse.DocsNameResponse;
import com.lendingkart.prakhar.lendingkartdemo.retrofitrequests.DocName;

import retrofit2.Call;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;
import retrofit2.http.Body;
import retrofit2.http.Multipart;
import retrofit2.http.POST;
import retrofit2.http.Part;
import retrofit2.http.PartMap;


    public interface APIInterface {

        String ENDPOINT = "https://app.xxxxxxxxx.com/";


        @POST("lkart/api/docs")
        Call<DocsNameResponse> DOCS_NAME_RESPONSE_CALL(@Body DocName docName);



        public static final Retrofit retrofit = new Retrofit.Builder()
                .baseUrl(APIInterface.ENDPOINT)
                .addConverterFactory(GsonConverterFactory.create())
                .build();

    }

позвоните так, где вы хотите

        String doc_name = "Loans/jdfjdanklnadkm;cnak_";
        APIInterface apiInterface = APIInterface.retrofit.create(APIInterface.class);


   Call<DocsNameResponse> DocsCall = apiInterface.DOCS_NAME_RESPONSE_CALL(new DocName(doc_name));
        DocsCall.enqueue(new Callback<DocsNameResponse>() {
            @Override
            public void onResponse(Call<DocsNameResponse> call, Response<DocsNameResponse> response) {
                Log.d("APIResult", String.valueOf(response.body().getData().get(3).getName()));
            }

            @Override
            public void onFailure(Call<DocsNameResponse> call, Throwable t) {
                Log.d("APIError", t.getMessage());
            }
        });

и два файла для запроса и ответа

Имя документа

public class DocName {
    private String name;

    public DocName(String name) {
        this.name = name;
    }

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

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

}

DocNameResponse Вы можете использовать http://www.jsonschema2pojo.org/ чтобы преобразовать ваш JSON в следующий письменный формат, выбрав SourceType: JSON и Annotation Style: GSON

import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;



import java.util.List;


public class DocsNameResponse {

    @SerializedName("message")
    @Expose
    private String message;
    @SerializedName("statusCode")
    @Expose
    private Integer statusCode;
    @SerializedName("data")
    @Expose
    private List<Datum> data = null;
    @SerializedName("list")
    @Expose
    private Object list;
    @SerializedName("cscStatus")
    @Expose
    private Boolean cscStatus;
    @SerializedName("status")
    @Expose
    private Object status;
    @SerializedName("eligibleStatus")
    @Expose
    private Boolean eligibleStatus;
    @SerializedName("pwd")
    @Expose
    private Object pwd;
    @SerializedName("uname")
    @Expose
    private Object uname;
    @SerializedName("assignedToList")
    @Expose
    private Object assignedToList;

    /**
     * @return The message
     */
    public String getMessage() {
        return message;
    }

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

    /**
     * @return The statusCode
     */
    public Integer getStatusCode() {
        return statusCode;
    }

    /**
     * @param statusCode The statusCode
     */
    public void setStatusCode(Integer statusCode) {
        this.statusCode = statusCode;
    }

    /**
     * @return The data
     */
    public List<Datum> getData() {
        return data;
    }

    /**
     * @param data The data
     */
    public void setData(List<Datum> data) {
        this.data = data;
    }

    /**
     * @return The list
     */
    public Object getList() {
        return list;
    }

    /**
     * @param list The list
     */
    public void setList(Object list) {
        this.list = list;
    }

    /**
     * @return The cscStatus
     */
    public Boolean getCscStatus() {
        return cscStatus;
    }

    /**
     * @param cscStatus The cscStatus
     */
    public void setCscStatus(Boolean cscStatus) {
        this.cscStatus = cscStatus;
    }

    /**
     * @return The status
     */
    public Object getStatus() {
        return status;
    }

    /**
     * @param status The status
     */
    public void setStatus(Object status) {
        this.status = status;
    }

    /**
     * @return The eligibleStatus
     */
    public Boolean getEligibleStatus() {
        return eligibleStatus;
    }

    /**
     * @param eligibleStatus The eligibleStatus
     */
    public void setEligibleStatus(Boolean eligibleStatus) {
        this.eligibleStatus = eligibleStatus;
    }

    /**
     * @return The pwd
     */
    public Object getPwd() {
        return pwd;
    }

    /**
     * @param pwd The pwd
     */
    public void setPwd(Object pwd) {
        this.pwd = pwd;
    }

    /**
     * @return The uname
     */
    public Object getUname() {
        return uname;
    }

    /**
     * @param uname The uname
     */
    public void setUname(Object uname) {
        this.uname = uname;
    }

    /**
     * @return The assignedToList
     */
    public Object getAssignedToList() {
        return assignedToList;
    }

    /**
     * @param assignedToList The assignedToList
     */
    public void setAssignedToList(Object assignedToList) {
        this.assignedToList = assignedToList;
    }


    public class Datum {

        @SerializedName("id")
        @Expose
        private Object id;
        @SerializedName("name")
        @Expose
        private String name;
        @SerializedName("applicationId")
        @Expose
        private Object applicationId;
        @SerializedName("userId")
        @Expose
        private Object userId;
        @SerializedName("documentName")
        @Expose
        private String documentName;
        @SerializedName("documentType")
        @Expose
        private Object documentType;
        @SerializedName("freshloan")
        @Expose
        private Object freshloan;

        /**
         * @return The id
         */
        public Object getId() {
            return id;
        }

        /**
         * @param id The id
         */
        public void setId(Object id) {
            this.id = id;
        }

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

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

        /**
         * @return The applicationId
         */
        public Object getApplicationId() {
            return applicationId;
        }

        /**
         * @param applicationId The applicationId
         */
        public void setApplicationId(Object applicationId) {
            this.applicationId = applicationId;
        }

        /**
         * @return The userId
         */
        public Object getUserId() {
            return userId;
        }

        /**
         * @param userId The userId
         */
        public void setUserId(Object userId) {
            this.userId = userId;
        }

        /**
         * @return The documentName
         */
        public String getDocumentName() {
            return documentName;
        }

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

        /**
         * @return The documentType
         */
        public Object getDocumentType() {
            return documentType;
        }

        /**
         * @param documentType The documentType
         */
        public void setDocumentType(Object documentType) {
            this.documentType = documentType;
        }

        /**
         * @return The freshloan
         */
        public Object getFreshloan() {
            return freshloan;
        }

        /**
         * @param freshloan The freshloan
         */
        public void setFreshloan(Object freshloan) {
            this.freshloan = freshloan;
        }

    }

}
person Prakhar1001    schedule 07.12.2016

Вот пример.

Коротко:

Gson gson = new GsonBuilder()
     .registerTypeAdapter(MyClass.class, new MyClassTypeAdapter())
     .create();

Retrofit retrofit = new Retrofit.Builder()  
     .baseUrl("https://api.github.com")
     .addConverterFactory(GsonConverterFactory.create(gson))
     .build();
person Andrey    schedule 10.12.2018