使GSON将数字反序列化为整数或双精度数

Mer*_*Hul 7 java json gson

我对GSON很难过.

我有一个简单的JSON,我想反序列化为Map<String,Object>.

我真的很直观,123应该被解析为int(或long),123.4应该被解析为float(或double).

另一方面,GSON一直创造双打.

我可以告诉GSON不要一直滥用双倍?

我的实际代码:

Type mapType = new TypeToken<Map<String, Object>>() {}.getType();
GSON gson = new Gson();
Map<String, Object> map = gson.fromJson(someString, mapType);
Run Code Online (Sandbox Code Playgroud)

g00*_*ur3 6

以下代码编译和工作:

package test;

import java.lang.reflect.Type;
import java.text.NumberFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

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.JsonParseException;
import com.google.gson.reflect.TypeToken;

public class Main {

    public static void main(String[] args) {
        GsonBuilder builder = new GsonBuilder();
        builder.registerTypeAdapter(Object.class, new MyObjectDeserializer());
        Gson gson = builder.create();
        String array = "[1, 2.5, 4, 5.66]";

        Type objectListType = new TypeToken<ArrayList<Object>>() {}.getType();
        List<Object> obj = gson.fromJson(array, objectListType);

        System.out.println(Arrays.toString(obj.toArray()));
    }

    public static class MyObjectDeserializer implements JsonDeserializer<Object> {

        public Object deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) 
            throws JsonParseException {

          Number num = null;
          try {
              num = NumberFormat.getInstance().parse(json.getAsJsonPrimitive().getAsString());
          } catch (Exception e) {
              //ignore
          }
          if (num == null) {
              return context.deserialize(json, typeOfT);
          } else {
              return num;
          }
      }
    }

}
Run Code Online (Sandbox Code Playgroud)

我的解决方案将首先尝试将字符串解析为数字,如果失败,它将让标准 Gson 反序列化器完成工作。

如果您需要一个不是特定于语言环境的数字解析器,请使用此方法来解析一个数字:

private static Number parse(String str) {
    Number number = null;
    try {
        number = Float.parseFloat(str);
    } catch(NumberFormatException e) {
    try {
        number = Double.parseDouble(str);
    } catch(NumberFormatException e1) {
        try {
            number = Integer.parseInt(str);
        } catch(NumberFormatException e2) {
            try {
                number = Long.parseLong(str);
            } catch(NumberFormatException e3) {
                throw e3;
            }       
        }       
    }       
}
    return number;
}
Run Code Online (Sandbox Code Playgroud)


may*_*ayr 1

混合这样的类型(整数与双精度数)并不是一个好方法。由于您使用对象作为类型,因此您将无法从映射中同时获取整数和双精度数。Gson 决定哪种类型更适合 Object。在您的情况下,它是 Double ,因为所有值都可以是双倍,但所有值不能是整数。

如果您确实需要混合类型,请尝试使用 Number 类而不是 Object。例子:

public static void main(String[] args){
        String array = "[1, 2.5, 4, 5.66]";
        Gson gson = new Gson();

        Type type = new TypeToken<ArrayList<Number>>() {}.getType();
        List<Number> obj = gson.fromJson(array, type);

        System.out.println(Arrays.toString(obj.toArray()));
    }
Run Code Online (Sandbox Code Playgroud)

输出:[1, 2.5, 4, 5.66]

虽然这样:

public static void main(String[] args){
    String array = "[1, 2.5, 4, 5.66]";
    Gson gson = new Gson();

    Type type = new TypeToken<ArrayList<Object>>() {}.getType();
    List<Object> obj = gson.fromJson(array, type);

    System.out.println(Arrays.toString(obj.toArray()));
}
Run Code Online (Sandbox Code Playgroud)

将为您提供输出:[1.0, 2.5, 4.0, 5.66]