GSON针对特定字段动态将Integer值转换为Boolean值

时间:2019-03-10 20:03:46

标签: java android json serialization gson

如何处理获得相同名称但不同类型的字段?我在同一个请求中有时从API获得整数值,有时是布尔值。我想知道让Json这样的人时该如何处理。我创建了类型适配器,但是它不起作用

我考虑过创建不同的POJO类。但是,这个问题并不只是一个请求。因此,我不喜欢创建POJO。顺便说一句,我看到了类似的问题,但是并不能解决我的问题。

{
  "name" : "john doe",
  "isValid" : true 
}

有时候我会理解

{
  "name" : "john doe",
  "isValid" : 1 
}

获取整数时出现意外的json异常

class XModel{
    private boolean isValid;
    ...
    ...
}

我想为每个请求返回一个布尔值。有人知道如何解决这个问题吗?

编辑: 我想通过类型适配器

阻止instanceOf关键字

解决方案:@MichałZiober的回复对我有用。

class BooleanJsonDeserializer implements JsonDeserializer<Boolean> {

    private final Set<String> TRUE_STRINGS = new HashSet<>(Arrays.asList("true", "1", "yes"));

    @Override
    public Boolean deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
        System.out.println(json);
        JsonPrimitive jsonPrimitive = json.getAsJsonPrimitive();
        if (jsonPrimitive.isBoolean()) {
            return jsonPrimitive.getAsBoolean();
        } else if (jsonPrimitive.isNumber()) {
            return jsonPrimitive.getAsNumber().intValue() == 1;
        } else if (jsonPrimitive.isString()) {
            return TRUE_STRINGS.contains(jsonPrimitive.getAsString().toLowerCase());
        }

        return false;
    }
}

2 个答案:

答案 0 :(得分:3)

如果XModel类不大,则可以按如下所示编写自定义反序列化器,以控制传入元素:

import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.JsonDeserializationContext;
import com.google.gson.JsonDeserializer;
import com.google.gson.JsonElement;
import com.google.gson.JsonObject;
import com.google.gson.JsonParseException;
import com.google.gson.JsonPrimitive;

import java.io.File;
import java.io.FileReader;
import java.lang.reflect.Type;
import java.util.Arrays;
import java.util.HashSet;
import java.util.Set;

public class GsonApp {

    public static void main(String[] args) throws Exception {
        File jsonFile = new File("./resource/test.json").getAbsoluteFile();

        Gson gson = new GsonBuilder()
                .registerTypeAdapter(XModel.class, new XModelJsonDeserializer())
                .create();

        System.out.println(gson.fromJson(new FileReader(jsonFile), XModel.class));
    }
}

class XModelJsonDeserializer implements JsonDeserializer<XModel> {

    private final Set<String> TRUE_STRINGS = new HashSet<>(Arrays.asList("true", "1", "yes"));

    @Override
    public XModel deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
        XModel response = new XModel();
        JsonObject jsonResponse = (JsonObject) json;
        response.setName(jsonResponse.get("name").getAsString());
        // other fields

        JsonElement dataElement = jsonResponse.get("isValid");
        if (dataElement.isJsonNull()) {
            response.setValid(false);
        } else if (dataElement.isJsonPrimitive()) {
            JsonPrimitive jsonPrimitive = dataElement.getAsJsonPrimitive();
            if (jsonPrimitive.isBoolean()) {
                response.setValid(jsonPrimitive.getAsBoolean());
            } else if (jsonPrimitive.isNumber()) {
                response.setValid(jsonPrimitive.getAsNumber().intValue() == 1);
            } else if (jsonPrimitive.isString()) {
                response.setValid(TRUE_STRINGS.contains(jsonPrimitive.getAsString()));
            }
            System.out.println("Json data is primitive: " + dataElement.getAsString());
        } else if (dataElement.isJsonObject() || dataElement.isJsonArray()) {
            response.setValid(true); //?!?!
        }

        return response;
    }
}

对于低于JSON的有效载荷:

{
  "name" : "john doe",
  "isValid" : true
}

程序上方的照片:

Json data is primitive: true
XModel{name='john doe', isValid=true}

对于JSON有效载荷:

{
  "name" : "john doe",
  "isValid" : 1
}

打印:

Json data is primitive: 1
XModel{name='john doe', isValid=true}

您的模型很清晰,因为所有工作都在反序列化器级别完成。

稍微精确一点的解决方案是仅序列化primitive。假设模型如下所示:

class XModel {

    private String name;

    @JsonAdapter(value = BooleanJsonDeserializer.class)
    private boolean isValid;

    // getters, setters
}

和我们的BooleanJsonDeserializer反序列化器如下所示:

class BooleanJsonDeserializer implements JsonDeserializer<Boolean> {

    private final Set<String> TRUE_STRINGS = new HashSet<>(Arrays.asList("true", "1", "yes"));

    @Override
    public Boolean deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
        System.out.println(json);
        JsonPrimitive jsonPrimitive = json.getAsJsonPrimitive();
        if (jsonPrimitive.isBoolean()) {
            return jsonPrimitive.getAsBoolean();
        } else if (jsonPrimitive.isNumber()) {
            return jsonPrimitive.getAsNumber().intValue() == 1;
        } else if (jsonPrimitive.isString()) {
            return TRUE_STRINGS.contains(jsonPrimitive.getAsString().toLowerCase());
        }

        return false;
    }
}

您只需在模型中使用此适配器注释每个boolean属性,即可处理1True

答案 1 :(得分:1)

我认为执行此映射并不容易,但是以下内容可能会有所帮助。

public void setIsValid(Object isValid) {
    String isValidString = String.valueOf(isValid).replace("0", "false").replace("1", "true");
    return Boolean.valueOf(isValidString);
}