使用 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。
作为我正在创建的 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。