如何处理获得相同名称但不同类型的字段?我在同一个请求中有时从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;
}
}
答案 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
属性,即可处理1
,True
等
答案 1 :(得分:1)
我认为执行此映射并不容易,但是以下内容可能会有所帮助。
public void setIsValid(Object isValid) {
String isValidString = String.valueOf(isValid).replace("0", "false").replace("1", "true");
return Boolean.valueOf(isValidString);
}