使用 JsonReader 读取动态 json 文件

Using JsonReader to read dynamic json files

作为我正在创建的 API 的一部分,我允许配置规范(可以是任何存储格式,只有一种实现是 Json)。作为其中的一部分,我的代码将不知道配置的真正含义。 我正在使用 Gson 库来读取 json 实现的配置,但是我在如何使用它处理数字方面遇到了障碍。 我当前的代码使用递归来读取内部对象,并包含以下内容:

private JsonConfigurationSection readObject(JsonReader in) throws IOException {
    JsonConfigurationSection section = new JsonConfigurationSection();
    in.beginObject();
    while (in.peek() != JsonToken.END_OBJECT) {
        String name = in.nextName();
        switch (in.peek()) {
            case BEGIN_OBJECT: {
                section._internal.put(name, readObject(in));
            }
            break;
            case BEGIN_ARRAY: {
                in.beginArray();
                List<String> array = new LinkedList<>();
                while (in.peek() != JsonToken.END_ARRAY) {
                    array.add(in.nextString());
                }
                in.endArray();
                section._internal.put(name, array);
            }
            break;
            case BOOLEAN: {
                boolean next = in.nextBoolean();
                section._internal.put(name, next);
            }
            break;
            case NUMBER: {
                //read the next number, whether long, int, or double    
                section._internal.put(name, next);
            }
            break;
            case STRING: {
                String next = in.nextString();
                section._internal.put(name, next);
            }
            break;
        }
    }
    in.endObject();
}

JsonConfigurationSection class 只是 Map 的包装器:

class JsonConfigurationSection implements ConfigurationSection {

    final Map<String, Object> _internal = new TreeMap<>();

    //methods being inherited, just getters for data from the map
}

一个配置示例可能是

{
  "server": {
    "ip": "127.0.0.1",
    "port": 3306
  }
  "someval": 33.4
}

出现的问题是 JsonReader 只为 "Number" 提供下一个标记,然后是用于 long、double 和 int 的特定 getter。

在不丢失任何数据并使用 "best" 存储的情况下获取该数字的最佳方法是什么? (我愿意放弃多头,但宁愿看看我是否可以保持它们的一致性)

抱歉把它作为答案。没有足够的声誉来添加评论。

所以。就像想法一样。您可以扩展 java.lang.Number 并在 String.

中保留值 "as is"

玩了一会儿之后,似乎最好的方法是只使用 nextString 并使用 BigDecimal,正如评论所建议的那样:

 String line = in.nextString();
 BigDecimal decimal = new BigDecimal(line);
 try {
     section._internal.put(name, decimal.intValueExact());
 } catch (ArithmeticException e) {
     try {
         section._internal.put(name, decimal.longValueExact());
     } catch (ArithmeticException ex) {
         section._internal.put(name, decimal.doubleValue());
     }
}

这实际上只是检查该值是否适合 int(这是 3 种类型中最有限的类型),然后是 long,如果失败,则保留为 double。