【问题标题】:Convert nested JSON Keys to UPPERCASE将嵌套的 JSON 键转换为大写
【发布时间】:2019-03-23 09:54:50
【问题描述】:

我想在 Java 中将 JSON 字符串/对象的所有键转换为大写。 JSON 可以嵌套。

我尝试在 GsonBuilder 中设置 FieldNamingPolicy.UPPER_CAMEL_CASE 但我猜这仅适用于 String to JAVA Object 而不适用于 String to String。

 String payload = "{\"key\" : {\"key1\" : \"value1\",\"key2\" : \"value2\"}}";
 GsonBuilder gsonBuilder = new GsonBuilder();
 gsonBuilder.registerTypeAdapterFactory(myCustomTypeAdapterFactory);
 gsonBuilder.setFieldNamingPolicy(FieldNamingPolicy.UPPER_CAMEL_CASE);
 Gson gson = gsonBuilder.create();
 Map mapDeserialized = gson.fromJson(payload, Map.class);

 System.out.println("Map " + mapDeserialized);

通过 JACKSON 可以使用自定义 TypeAdapterFactory 提供其他解决方案,但这些解决方案仅适用于一层,不适用于嵌套。

{"key" : { "key1" : "value1", "key2" : "value2" }}

{"KEY" : { "KEY1" : "value1", "KEY2" : "value2" }}

【问题讨论】:

标签: java json gson


【解决方案1】:

正如您所说,FieldNamingPolicy 仅适用于 bean 字段而不适用于映射键。但是 UPPER_CAMEL_CASE 不是您想要的,它是首字母大写的驼峰式大小写(SometingLikeThis)。你必须实现你自己的反序列化器来为你做到这一点:

import com.google.gson.*;
import com.google.gson.reflect.TypeToken;

import java.lang.reflect.Type;
import java.util.HashMap;
import java.util.Map;

public class UpperCaseAdapter implements JsonSerializer<Map<String, Object>>, JsonDeserializer<Map<String, Object>> {
    public static final Type TYPE = new TypeToken<Map<String, Object>>() {}.getType();

    @Override
    public JsonElement serialize(Map<String, Object> src, Type typeOfSrc, JsonSerializationContext context) {
        // TODO implement serialization if needed
        return null;
    }

    @Override
    public Map<String, Object> deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException {
        Map<String, Object> map = new HashMap<>();
        for (Map.Entry<String, JsonElement> entry : json.getAsJsonObject().entrySet()) {
            Object value = null;
            if (entry.getValue().isJsonPrimitive()) {
                value = entry.getValue().getAsString();
            } else if (entry.getValue().isJsonObject()) {
                value = context.deserialize(entry.getValue(), TYPE); // deserialize the object using the same type
            } else if (entry.getValue().isJsonArray()) {
                // TODO implement deserailization of array
            } else if (entry.getValue().isJsonNull()) {
                // skip nulls
                continue;
            }
            map.put(entry.getKey().toUpperCase(), value); //toUpperCase() is what we want
        }
        return map;
    }
}

然后您可以使用适配器:

        String payload = "{\"key\" : {\"key1\" : \"value1\",\"key2\" : \"value2\"}, \"key3\": \"value\"}";
        Gson gson = new GsonBuilder()
                .registerTypeAdapter(UpperCaseAdapter.TYPE, new UpperCaseAdapter())
                .create();
        Map<String, Object> mapDeserialized = gson.fromJson(payload, UpperCaseAdapter.TYPE);

        System.out.println("Map " + mapDeserialized);

输出是: Map {KEY3=value, KEY={KEY2=value2, KEY1=value1}}

【讨论】:

  • 此解决方案是否适用于任何级别的嵌套 JSON?
  • 当然可以,你可以试试。有一个递归关心任何嵌套的 lvl。但是如果您希望在 json 中成为一个数组,则必须改进实现。
  • 非常感谢你。我再次为 arrya 的每个元素调用了上下文,它适用于 4 个级别。
  • 不客气。只有一件事 - 确保数组的元素是 json 对象,如果元素是原始数组或另一个数组,那么 context.deserialze 的调用将不适用于 TYPE。数组反序列化最安全的方法是实现另一个适配器,它期望 json 元素是 json 数组并遍历元素,如果元素是对象,则为 map TYPE 调用反序列化,如果元素为数组,则调用反序列化数组类型,如果元素是原始元素,则将元素作为字符串添加到数组中。
猜你喜欢
  • 2020-05-08
  • 2020-10-26
  • 1970-01-01
  • 1970-01-01
  • 2018-01-07
  • 2020-10-28
  • 1970-01-01
  • 2017-06-01
相关资源
最近更新 更多