为什么Gson将整数解析为双精度值?

2022-09-02 13:44:32

一个复杂的json字符串,我想把它转换为map,我有一个问题。

请看这个简单的测试:

public class Test {

    @SuppressWarnings("serial")
    public static void main(String[] args) {
        Map<String, Object> hashMap = new HashMap<String, Object>();
        hashMap.put("data", "{\"rowNum\":0,\"colNum\":2,\"text\":\"math\"}");

        Map<String,Object> dataMap = JsonUtil.getGson().fromJson(
                hashMap.get("data").toString(),new TypeToken<Map<String,Object>>() {}.getType());

        System.out.println(dataMap.toString());

    }
}

结果:
控制台打印:
Int转换为Double;
为什么gson会更改类型,我该如何修复它?{rowNum=0.0, colNum=2.0, text=math}


答案 1

Gson是一个简单的解析器。如果要将数据解析为 ,它将始终使用 Double 作为默认数字类型。Object

有关详细信息,请查看此问题:如何防止 Gson 将整数表示为浮点数

我建议你使用Jackson Mapper。Jackson 区分类型,即使您正在解析对象:

  • "2"Integer
  • "2.0"Double

下面是一个示例:

Map<String, Object> hashMap = new HashMap<String, Object>();
hashMap.put("data", "{\"rowNum\":0,\"colNum\":2,\"text\":\"math\"}");
ObjectMapper mapper = new ObjectMapper();
TypeReference<HashMap<String, Object>> typeRef = new TypeReference<HashMap<String, Object>>() {};

HashMap<String, Object> o = mapper.readValue(hashMap.get("data").toString(), typeRef);

专家:

<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-core</artifactId>
    <version>2.9.0</version>
</dependency>

<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.9.0</version>
</dependency>

答案 2

JSON不像Java那样区分不同类型的数字。它将所有类型的数字视为单一类型。

将这些数字解析为 a 是 Gson 库的实现细节。当它遇到 JSON 编号时,它默认将其解析为 .DoubleDouble

与其使用 ,不如定义一个封装 JSON 结构的所有字段的 POJO。这使得之后访问数据变得更加容易,并且数字会自动解析为.MapInteger

class Cell {
    private Integer rowNum;
    private Integer colNum;
    private String text;
}

public static void main(String[] args) throws Exception {
    Map<String, Object> hashMap = new HashMap<String, Object>();
    hashMap.put("data", "{\"rowNum\":0,\"colNum\":2,\"text\":\"math\"}");

    Cell cell = new Gson().fromJson(hashMap.get("data").toString(), Cell.class);
    System.out.println(cell);
}