当前位置: 首页 > 知识库问答 >
问题:

Jackson json to map和camelcase键名

吉玉石
2023-03-14

我想通过杰克逊库将json转换为包含camelCase密钥的地图……比如……

{
    "SomeKey": "SomeValue",
    "AnotherKey": "another value",
    "InnerJson" : {"TheKey" : "TheValue"}
}

到这个。。。

{
    "someKey": "SomeValue",
    "anotherKey": "another value",
    "innerJson" : {"theKey" : "TheValue"}
}

我的代码...

public Map<String, Object> jsonToMap(String jsonString) throws IOException
{
    ObjectMapper mapper=new ObjectMapper();
    mapper.setPropertyNamingStrategy(PropertyNamingStrategy.CAMEL_CASE_TO_LOWER_CASE_WITH_UNDERSCORES);
    return mapper.readValue(jsonString,new TypeReference<Map<String, Object>>(){});
}

但这行不通...甚至其他属性命名策略也不适用于 json...如。。。

{
    "someKey": "SomeValue"
}

mapper.setPropertyNamingStrategy(new PropertyNamingStrategy.PascalCaseStrategy())

{
    "SomeKey": "SomeValue"
}

如何通过jackson获得camelCase地图键名...或者我应该手动循环映射和转换关键点还是有一些其他的方法???

提前感谢...

共有3个答案

呼延曜灿
2023-03-14

下面将使用任何“case格式”的键将json转换为使用camelCased键:

/**
 * Convert all property keys of the specified JSON to camelCase
 */
public static String toJsonWithCamelCasedKeys(String json) {
    ObjectMapper objectMapper = new ObjectMapper();
    objectMapper.registerModule(new SimpleModule()
      .addKeySerializer(String.class, new JsonSerializer<>() {
          @Override
          public void serialize(String value, JsonGenerator gen, SerializerProvider serializers) throws IOException {
              String key = CaseUtil.toCamelCase(value);
              gen.writeFieldName(key);
          }
      })
    );

    try {
        Map<String, Object> jsonMap = objectMapper.readValue(json, new TypeReference<>() {});
        return objectMapper.writeValueAsString(jsonMap);
    } catch (Exception e) {
        throw new JsonException("Error transforming JSON", e);
    }
}

… CaseUtil实现可能是这样的:

import java.util.Arrays;
import java.util.stream.Collectors;

public class CaseUtil {

    public static String toCamelCase(String s) {
        if (s == null) {
            return null;
        }
        else if (s.isBlank()) {
            return "";
        }

        return decapitaliseFirstLetter(
          String.join("", Arrays.stream(s.split("[-_\\s]"))
          .map(CaseUtil::capitaliseFirstLetter)
          .collect(Collectors.toList()))
        );
    }

    private static String capitaliseFirstLetter(String s) {
        return (s.length() > 0)
          ? s.substring(0, 1).toUpperCase() + s.substring(1)
          : s;
    }

    private static String decapitaliseFirstLetter(String s) {
        return (s.length() > 0)
          ? s.substring(0, 1).toLowerCase() + s.substring(1)
          : s;
    }

}

单元测试:

    @Test
    void jsonWithMiscCasedPropKeys_shouldConvertKeyToCamelCase() throws Exception {
        String inputJson =
            "{\"kebab-prop\": \"kebab\"," +
            "\"snake_prop\": \"snake\"," +
            "\"PascalProp\": \"pascal\"," +
            "\"camelCasedProp\": \"camel\"}";
        String expectedJson =
          "{\"kebabProp\": \"kebab\"," +
            "\"snakeProp\": \"snake\"," +
            "\"pascalProp\": \"pascal\"," +
            "\"camelCasedProp\": \"camel\"}";
        String actualJson = Json.toJsonWithCamelCasedKeys(inputJson);

        JSONAssert.assertEquals(expectedJson, actualJson, JSONCompareMode.LENIENT);
    }
公冶伟
2023-03-14

您始终可以迭代地图的键并更新它们。但是,如果您只对使用驼峰大小写键生成JSON感兴趣,可以考虑下面描述的方法。

您可以有一个自定义密钥序列化器。它将在将Map实例序列化为JSON时使用:

public class CamelCaseKeySerializer extends JsonSerializer<String> {

    @Override
    public void serialize(String value, JsonGenerator gen, SerializerProvider serializers)
                throws IOException, JsonProcessingException {

        String key = Character.toLowerCase(value.charAt(0)) + value.substring(1);
        gen.writeFieldName(key);
    }
}

然后执行以下操作:

String json = "{\"SomeKey\":\"SomeValue\",\"AnotherKey\":\"another value\",\"InnerJson\":"
            + "{\"TheKey\":\"TheValue\"}}";

SimpleModule simpleModule = new SimpleModule();
simpleModule.addKeySerializer(String.class, new CamelCaseKeySerializer());

ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(simpleModule);

Map<String, Object> map = mapper.readValue(json, 
                                          new TypeReference<Map<String, Object>>() {});

String camelCaseJson = mapper.writerWithDefaultPrettyPrinter().writeValueAsString(map);

输出将是:

{
  "someKey" : "SomeValue",
  "anotherKey" : "another value",
  "innerJson" : {
    "theKey" : "TheValue"
  }
}

使用这种方法,< code>Map的键将不会是camel大小写。但是它会给你想要的输出。

施利
2023-03-14

由于您使用的是地图/字典,而不是将JSON数据绑定到POJOs(与JSON数据匹配的显式Java类),因此属性命名策略不适用:

类PropertyNamingStrategy…定义了JSON属性的名称(“外部名称”)是如何从POJO方法和字段的名称派生出来的(“内部名称”)

因此,您必须首先使用Jackson解析数据,然后迭代结果并转换键。

像这样更改代码:

public Map<String, Object> jsonToMap(String jsonString) throws IOException
{
    ObjectMapper mapper=new ObjectMapper();
    mapper.setPropertyNamingStrategy(PropertyNamingStrategy.CAMEL_CASE_TO_LOWER_CASE_WITH_UNDERSCORES);
    Map<String, Object> map = mapper.readValue(jsonString,new TypeReference<Map<String, Object>>(){});
    return convertMap(map);
}

并添加这些方法:

public String mapKey(String key) {
    return Character.toLowerCase(key.charAt(0)) + key.substring(1);
}

public Map<String, Object> convertMap(Map<String, Object> map) {
    Map<String, Object> result = new HashMap<String, Object>();
    for (Map.Entry<String, Object> entry : map.entrySet()) {
        String key = entry.getKey();
        Object value = entry.getValue();
        result.put(mapKey(key), convertValue(value));
    }
    return result;
}

public convertList(Lst<Object> list) {
    List<Object> result = new ArrayList<Object>();
    for (Object obj : list) {
        result.add(convertValue(obj));
    }
    return result;
}

public Object covertValue(Object obj) {
    if (obj instanceof Map<String, Object>) {
        return convertMap((Map<String, Object>) obj);
    } else if (obj instanceof List<Object>) {
        return convertList((List<Object>) obj);
    } else {
        return obj;
    }
}
 类似资料:
  • 问题内容: AngularJS中用$前缀为提供者的约定是什么?我应该在自己的代码中为所有自定义服务添加前缀吗? 看起来angular附带的所有东西都带有前缀服务,例如。但是,在大多数文章中都没有以控制器为前缀。此外,所有角度代码都带有以中命名的服务,但是我也在许多在线博客中看到过。约定是哪一个? 问题答案: docs声明了内部服务的约定,但同时也声明您不应为自己的服务使用此约定以减少命名冲突。 h

  • 我有Spring4+Hibernate4.3+PSQL9.5应用程序,遇到了一些camelcase字段转换的问题。

  • 我正在尝试使用PropertyName策略。SNAKE_CASE转换地图 我的班级示例: 非常感谢。

  • 问题内容: 我找到了一个出色的RegEx来提取camelCase或TitleCase表达的一部分。 它按预期工作: value -> value camelValue -> camel / Value TitleValue -> Title / Value 例如,使用Java: 我的问题是在某些情况下它不起作用: 情况1:VALUE-> V / A / L / U / E 情况2:eclipseR

  • 问题内容: 我想要达到的目标是这样的: 所以我搜索并找到了这个完美的正则表达式: 作为下一个逻辑步骤,我尝试了: 为什么这不起作用,如何从python中的链接问题中获得结果? 编辑:解决方案摘要 我用一些测试用例测试了所有提供的解决方案: 总而言之,您可以说@kalefranz的解决方案与问题不符(请参阅最后一种情况),而@casimir et hippolyte的解决方案占用了一个空格,因此违反

  • 当使用protoc gen go从proto文件生成go代码时,json的密钥名将保留为proto文件中指定的密钥。在官方字段中,建议使用snake案例。https://developers.google.com/protocol-buffers/docs/style 但是,我希望json的关键名称是蛇案。 当我检查protoc gen go的生成器代码时,我当然会按原样设置字段名。 这将是我的理