- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我开始使用 Jackson ObjectMapper
将 JSON String
解析为 MongoDB BasicDBObject
。
在 readValue
期间,我希望 Jackson 将字段名称中的任何保留字符替换为替代字符
在 writeValue
期间,我希望 Jackson 进行反向替换
这样做的原因是 Mongo DB 不支持 .
或 $
作为 JSON 中键名称的一部分,而我有旧数据,但我不支持太热衷于转换,这样我就可以在没有特殊字符映射的情况下坚持它..
我正在寻找正确的集成点,在该集成点上我可以覆盖 readValue
期间生成的字段名称,以及将对象序列化回字符串时使用的集成点 (writeValue
)。
我查看了 SimpleModule.addKeySerializer
和 SimpleModule.addKeyDeserializer
,并实现了一个简单的映射。然而,这似乎只适用于我的嵌套 JSON 对象的第一层。任何嵌套对象的名称似乎都没有正确映射。
覆盖字段名称映射方式的正确集成点是什么?
最佳答案
我应该在序言中提到,我不是 Jackson 方面的专家,但我看到两个可能的地方可以集成此类功能:
第一种方法可能有点繁琐,但您可以编写自定义 serializer和 deserializer (或多个。本例中我使用了其中一个)来处理键名中的字符替换。
举个例子,我正在使用一些包含一些句点和美元符号的 JSON,如下所示:
{\"username.\":\"testuser\",\"id\":\"12345\",\"$role\":{\"role\":\"admin\",\"roleId$\":\"999\"}}
并将替换“.”与“_”,“$”与“|”,然后再返回。我不太确定如何进一步浓缩这一点,所以请原谅这篇长文章。
使用类:
public class User {
private final String username;
private final String id;
private final RoleInfo role;
public User(String username, String id, RoleInfo role) {
this.username = username;
this.id = id;
this.role = role;
}
...getters omitted
}
public class RoleInfo {
private final String role;
private final String roleId;
public RoleInfo(String role, String roleId) {
this.role = role;
this.roleId = roleId;
}
...getters omitted
}
还有序列化器/反序列化器(假设是最新版本的 Jackson,您可能希望比我在本例中更仔细地处理节点的处理方式):
public class MyCustomDeserializer extends JsonDeserializer<User> {
private final String period;
private final String dollarSign;
public MyCustomDeserializer(final String period, final String dollarSign){
this.period = period;
this.dollarSign = dollarSign;
}
@Override
public User deserialize(JsonParser p, DeserializationContext ctxt)
throws IOException, JsonProcessingException {
final TreeNode node = p.getCodec().readTree(p);
final TextNode roleNode = (TextNode) node.get(this.dollarSign + "role").get("role");
final TextNode roleIdNode = (TextNode) node.get(this.dollarSign + "role").get("roleId" + this.dollarSign);
final String roleName = roleNode.asText();
final String roleId = roleIdNode.asText();
final RoleInfo roleInfo = new RoleInfo(roleName, roleId);
final TextNode usernameNode = (TextNode) node.get("username" + period);
final TextNode idNode = (TextNode) node.get("id");
final String username = usernameNode.asText();
final String userId = idNode.asText();
final User user = new User(username, userId, roleInfo);
return user;
}
}
public class MyCustomSerializer extends JsonSerializer<User> {
private final String period;
private final String dollarSign;
public MyCustomSerializer(final String period, final String dollarSign) {
this.period = period;
this.dollarSign = dollarSign;
}
@Override
public void serialize(User user, JsonGenerator gen,
SerializerProvider serializers) throws IOException,
JsonProcessingException {
gen.writeStartObject();
gen.writeStringField("username" + this.period, user.getUsername());
gen.writeStringField("id", user.getId());
gen.writeFieldName(this.dollarSign + "role");
gen.writeStartObject();
gen.writeStringField("role", user.getRole().getRole());
gen.writeStringField("roleId" + this.dollarSign, user.getRole()
.getRoleId());
gen.writeEndObject();
gen.writeEndObject();
}
}
序列化和反序列化:
final String testJson = "{\"username.\":\"testuser\",\"id\":\"12345\",\"$role\":{\"role\":\"admin\",\"roleId$\":\"999\"}}";
final ObjectMapper mapper = new ObjectMapper();
SimpleModule module = new SimpleModule();
module.addDeserializer(User.class, new MyCustomDeserializer(".", "$"));
module.addSerializer(User.class, new MyCustomSerializer("_", "|"));
mapper.registerModule(module);
final User user = mapper.readValue(testJson, User.class);
printUser(user);
final String convertedJson = mapper.writeValueAsString(user);
System.out.println(convertedJson);
System.out.println("");
final ObjectMapper mapper2 = new ObjectMapper();
SimpleModule module2 = new SimpleModule();
module2.addDeserializer(User.class, new MyCustomDeserializer("_", "|"));
module2.addSerializer(User.class, new MyCustomSerializer(".", "$"));
mapper2.registerModule(module2);
final User user2 = mapper2.readValue(convertedJson, User.class);
printUser(user2);
final String json2 = mapper2.writeValueAsString(user);
System.out.println(json2);
产品:
12345
testuser
admin
999
{"username_":"testuser","id":"12345","|role":{"role":"admin","roleId|":"999"}}
12345
testuser
admin
999
{"username.":"testuser","id":"12345","$role":{"role":"admin","roleId$":"999"}}
正如我所说,这相当繁琐,因此您也可以尝试创建一些 POJO 来为您进行映射。我不确定是否有更优雅的方法使用各种 json annotations and some POJOs 来解决这个问题,但我没有看到一种方法可以使用这种方法在一个 POJO 中使用两组替换字符进行序列化和反序列化。 DRCB's approach链接中的结果为我提供了序列化的两个属性。
我想您还可以考虑支持两组 POJOS,其中每个保留字符都在各自的注释中,并在您从各种来源读取/写入时在它们之间进行转换。类似于(伪代码):
ClassA{
@JsonProperty("$user")
private String user;
}
ClassB{
@JsonProperty("#user")
private String user;
}
ClassA a = ClassA.from(classB);
或者您可以结合这两种方法并编写自定义序列化程序,并使用具有多个 setter 注释的 POJO 进行反序列化。我认为解决方案会根据 JSON 和 POJO 的复杂性而有所不同。
关于java - Jackson JSON 键名称映射,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33598308/
我找不到关于 jackson 的ObjectMapper与其他映射器(例如dozer/mapStruct/modelMapping/etc)之间的区别的任何解释。所有文章都比较了dozer/mapSt
我正在使用Jackson来反序列化Kotlin数据类。我正在使用jackson-kotlin-module,但 jackson 却给我以下错误: Can not construct instance
我正在尝试将包从“com.fasterxml.jackson”重新定位到我自己的包(即“mypackage.com.fasterxml.jackson”),然后在我拥有的另一个 JAR 中使用它。 我
对于JSON对象,Subject: { "permissions":["foo", "bar"], ... } ...我想反序列化为: class Subject { priv
我正在使用 @JsonTypeInfo 和 @JsonSubTypes 来映射基于给定属性的解析子类。这是我想要解析的示例 JSON 的一个人为示例。 { "animals": [ { "
我们正在使用 dropwizard 版本 0.6.3。当我们尝试升级 0.7.0 版本时,我们在服务启动时收到此错误。 线程“main”中的异常 java.lang.VerifyError: clas
我正在尝试实现自定义解串器。因为我只想向默认反序列化器添加功能,所以我尝试在我的自定义反序列化器中存储默认反序列化器:我想使用默认反序列化 json,然后添加其他信息。 我正在尝试使用 BeanDes
我有一个这样的类(class): public class Person { private String name; public String getName(){ return
我有以下 Kotlin 数据类: data class TestObject( val boolField: Boolean, val stringField: Str
使用 Jackson 库,在 Eclipse 4.9.0 版本中出现以下错误 缺少工件 com.fasterxml.jackson.core:jackson-databind:bundle:2.9.6
我试图在我的应用程序中从azure实现keyvault,在为DefaultAzureCredentialBuilder()实现azure-identity:1.5.4 lib后,它会抛出链接错误,如下
我试图在我的应用程序中从azure实现keyvault,在为DefaultAzureCredentialBuilder()实现azure-identity:1.5.4 lib后,它会抛出链接错误,如下
我知道我们可以使用 Jackson MixIn 来重命名属性或忽略属性(参见示例 here )。但是可以添加属性吗? 添加的属性可以是: 一个常数(如版本号) 计算值(例如,如果源类具有 getWid
我有一个在 Wildfly 10 上运行的应用程序,它需要更新版本的 jackson。简单地更新 maven 依赖是行不通的。 Wildflys 自己的版本似乎干扰了... 有人有提示吗? 最佳答案
我在 Tomcat 休息应用程序中运行 Jersey 2.5.1 & Jackson。对于我最初将 POJO 简单地转换为 JSON 的用例,基本设置效果很好。集合很好地转换为 json 数组,如下所
有没有办法告诉 jackson 在序列化过程中忽略那些用非 jackson 注释注释的字段? 例如: @SomeAnnotation private String foo; 我知道有 jackson
我遇到了 jackson 序列化问题, map 中存在空值。显然,这是 Wildfly 9 使用的 Jackson 版本中的一个已知错误 ( https://issues.jboss.org/brow
给定一个像这样的 JSON 映射类: public class Person { @JsonProperty String getName() { ... } @JsonPro
如何让 Jackson 的 XMLMapper 在反序列化时读取根 xml 元素的名称? 我正在将输入 XML 反序列化为通用 Java 类、LinkedHashMap,然后反序列化为 JSON。我想
我对抽象类和 JSON 序列化和反序列化的对象引用有问题。抽象的问题如下所示: 我有一个由节点和边组成的图。每条边连接两个节点。节点可以是红色和绿色的。因此,有一个抽象类Node和两个派生类 RedN
我是一名优秀的程序员,十分优秀!