改造多种响应类型

Jum*_*mpa 6 java android json gson retrofit2

如何使用Retrofit2来解析这两种API响应?

好的响应(HTTP 200):

{
    "data": {
        "foo": "bar"
    }
}
Run Code Online (Sandbox Code Playgroud)

错误响应(HTTP 200):

{
    "error": {
        "foo": "bar"
    }
}
Run Code Online (Sandbox Code Playgroud)

我读过很多问题和教程,但我不知道该怎么做,我尝试过:

GsonBuilder gsonBuilder = new GsonBuilder();
gsonBuilder.registerTypeAdapterFactory(new ItemTypeAdapterFactory());
Gson gson = gsonBuilder.create();

final Retrofit retrofit = new Retrofit.Builder()
        .client(getOkHttpClient())
        .baseUrl(Constants.API_BASE_URL)
        .addConverterFactory(GsonConverterFactory.create(gson))
        .build();
Run Code Online (Sandbox Code Playgroud)

这是我的 ItemTypeAdapterFactory:

class ItemTypeAdapterFactory implements TypeAdapterFactory {

    public <T> TypeAdapter<T> create(final Gson gson, final TypeToken<T> type) {

        final TypeAdapter<T> delegate = gson.getDelegateAdapter(this, type);
        final TypeAdapter<JsonElement> elementAdapter = gson.getAdapter(JsonElement.class);

        return new TypeAdapter<T>() {

            public void write(JsonWriter out, T value) throws IOException {
                delegate.write(out, value);
            }

            public T read(JsonReader in) throws IOException {

                JsonElement jsonElement = elementAdapter.read(in);

                if (jsonElement.isJsonObject()) {
                    JsonObject jsonObject = jsonElement.getAsJsonObject();

                    // Data key
                    if (jsonObject.has(Constants.JSON_KEY_DATA)) {

                        JsonElement jsonData = jsonObject.get(Constants.JSON_KEY_DATA);

                        // Primitive
                        if (jsonData.isJsonPrimitive()) {
                            jsonElement = jsonData.getAsJsonPrimitive();
                        }
                        // JSON object
                        else if (jsonData.isJsonObject()) {
                            jsonElement = jsonData;
                        }
                        // JSON object array
                        else if (jsonData.isJsonArray()) {
                            jsonElement = jsonData.getAsJsonArray();
                        }
                    }
                }

                return delegate.fromJsonTree(jsonElement);
            }
        }.nullSafe();
    }
}
Run Code Online (Sandbox Code Playgroud)

但现在我不知道要在Retrofit2接口上声明的类型,在Call内部:

@GET("login")
Call<?> login(@Query(Constants.API_PARAM_TOKEN) String token);
Run Code Online (Sandbox Code Playgroud)

你能指出我正确的方向吗?

Qaz*_*han 9

在类似的情况下,我曾经使用过JsonObjectas 类型,所以你的函数将如下所示:

@GET("login")
Call<?> login(@Query(Constants.API_PARAM_TOKEN) String token);
Run Code Online (Sandbox Code Playgroud)

接下来,当您进行改造调用时,您将响应保留为字符串。因此,在您的 java 代码中,执行如下操作:

Call<JsonObject> call = RetrofitClient.getAPIService().login('YOUR_INPUT');
Data data = null;
Error error = null;
call.enqueue(new Callback<JsonObject>() {
            @Override
            public void onResponse(Call<JsonObject> call, Response<JsonObject> response) {
if(response.isSuccessfull()){
  String jsonString = response.body().toString();
  if(jsonString.contains("data:")){
       data = new Gson().fromJson(jsonString,Data.class);
  }else{
      error = new Gson().fromJson(jsonString,Error.class);
  }
}
        }

Run Code Online (Sandbox Code Playgroud)

在这里,我使用了DataError这两个类。他们是 POJO。所以数据可以看起来像这样:

Data.java:

public class Data implements Serializable{
 @SerializedName("foo")
    @Expose
    private Foo foo; // Foo is your desired data type 

}
Run Code Online (Sandbox Code Playgroud)

同样适用于Error. 因此,根据您的其余代码,进行必要的更改。祝你好运。