sha*_*era 3 java serialization json jackson
我有这个JSON反序列化:
{
"first-name": "Alpha",
"last-name": "Beta",
"gender": "m"
}
Run Code Online (Sandbox Code Playgroud)
我想将其序列化为2种不同的格式:
[一种]
{
"first-name": "Alpha",
"last-name": "Beta",
"gender": "m"
}
Run Code Online (Sandbox Code Playgroud)
[B]
{
"firstName": "Alpha",
"lastName": "Beta",
"gender": "m"
}
Run Code Online (Sandbox Code Playgroud)
我能够将其序列化为1种格式:仅[A]或[B]。这是将其序列化为[B]的代码:
public String firstName;
public String lastName;
public String gender;
@JsonProperty("firstName")
public String getFirstNameCC() {
return firstName;
}
@JsonProperty("first-name")
public void setFirstNameD(String firstName) {
this.firstName = firstName;
}
@JsonProperty("lastName")
public String getLastNameCC() {
return lastName;
}
@JsonProperty("last-name")
public void setLastNameD(String lastName) {
this.lastName = lastName;
}
public String getGender() {
return gender;
}
public void setGender(String gender) {
this.gender = gender;
}
Run Code Online (Sandbox Code Playgroud)
我在JsonView这里了解到http://www.baeldung.com/jackson-json-view-annotation(“5。自定义JSON视图”部分),但它只会更改其值。我想像上面的例子一样更改字段名称。谁能对此提供见解?
我不确定我是否能完全理解您的问题,但是据我所知,您可以执行类似的操作来实现不同的序列化。
创建一个自定义批注以保存所有可能的不同序列化选项:
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface CustomJsonProperty {
String propertyName();
String format();
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@interface List {
CustomJsonProperty[] value();
}
}
Run Code Online (Sandbox Code Playgroud)
相应地注释您的班级:
@JsonSerialize(using = CustomJsonPropertySerializer.class)
public class Bar {
@CustomJsonProperty.List({
@CustomJsonProperty(propertyName = "first-name", format = "A"),
@CustomJsonProperty(propertyName = "firstName", format = "B")
})
private String firstName;
@CustomJsonProperty.List({
@CustomJsonProperty(propertyName = "last-name", format = "A"),
@CustomJsonProperty(propertyName = "lastName", format = "B")
})
private String lastName;
@CustomJsonProperty.List({
@CustomJsonProperty(propertyName = "gender-x", format = "A"),
@CustomJsonProperty(propertyName = "gender", format = "B")
})
private String gender;
@JsonIgnore
private String format;
//getters & setters
}
Run Code Online (Sandbox Code Playgroud)
创建一个自定义序列化程序来解释您的新注释:
public class CustomJsonPropertySerializer extends JsonSerializer<Bar> {
@Override
public void serialize(Bar bar, JsonGenerator jsonGenerator, SerializerProvider serializerProvider)
throws IOException {
jsonGenerator.writeStartObject();
Field[] fields = bar.getClass().getDeclaredFields();
for (Field field : fields) {
field.setAccessible(true);
Object value = null;
try {
value = field.get(bar);
} catch (IllegalAccessException e) {
e.printStackTrace();
}
if (field.isAnnotationPresent(CustomJsonProperty.List.class)) {
CustomJsonProperty[] properties = field.getAnnotation(CustomJsonProperty.List.class).value();
CustomJsonProperty chosenProperty = null;
for (CustomJsonProperty c : properties) {
if (c.format().equalsIgnoreCase(bar.getFormat())) {
chosenProperty = c;
break;
}
}
if (chosenProperty == null) {
//invalid format given, use first format then
chosenProperty = properties[0];
}
jsonGenerator.writeStringField(chosenProperty.propertyName(), value.toString());
}
}
jsonGenerator.writeEndObject();
}
}
Run Code Online (Sandbox Code Playgroud)
现在,您可以考虑属性名称的不同格式来序列化对象:
public static void main(String[] args) throws IOException {
Bar bar1 = new Bar("first", "last", "m", "A");
Bar bar2 = new Bar("first", "last", "m", "B");
ObjectMapper mapper = new ObjectMapper();
String json1 = mapper.writeValueAsString(bar1);
String json2 = mapper.writeValueAsString(bar2);
System.out.println(json1);
System.out.println(json2);
}
Run Code Online (Sandbox Code Playgroud)
输出:
{"first-name":"first","last-name":"last","gender-x":"m"}
{"firstName":"first","lastName":"last","gender":"m"}
Run Code Online (Sandbox Code Playgroud)
当然,上述序列化程序仅适用于Bar对象,但是可以通过abstract String getFormat();在父类上使用继承并更改自定义序列化器以接受父类而不是Bar 来轻松解决。
也许有比创建自己的东西更简单的方法,但是我不知道。让我知道是否有不清楚的地方,我可以再次详细说明。
有一种更简单的方法可以做到这一点 - 创建一个使用“addMixin”函数的对象映射器。
要序列化的类:
Class YouWantToSerializeMe {
public String firstName;
public String lastName;
public String gender;
@JsonProperty("firstName")
public String getFirstNameCC() {
return firstName;
}
@JsonProperty("lastName")
public String getLastNameCC() {
return lastName;
}
}
Run Code Online (Sandbox Code Playgroud)
现在,要使用内置字段名称和自定义字段名称进行序列化,您可以执行以下操作:
Class DoTheSerializing {
String serializeNormally(YouWantToSerializeMe me) {
ObjectMapper objectMapper = new ObjectMapper();
ObjectWriter objectWriter = objectMapper.writer();
return objectWriter(me)
}
String serializeWithMixin(YouWantToSerializeMe me) {
ObjectMapper objectMapper = new ObjectMapper();
ObjectWriter objectWriter = objectMapper
.addMixIn(YouWantToSerializeMe.class, MyMixin.class)
.writer();
return objectWriter(me)
}
interface MyMixin {
@JsonProperty("first-name")
public String getFirstNameCC();
@JsonProperty("last-name")
public String getLastNameCC();
}
}
Run Code Online (Sandbox Code Playgroud)
这使用类中的嵌入式接口来保持事物的本地化。您可以围绕此进行大量优化,例如创建静态 ObjectMapper 和加载/卸载 mixin。
用界面作为“模板”来控制映射功能确实很强大。您可以在字段和类级别添加内容。
| 归档时间: |
|
| 查看次数: |
2537 次 |
| 最近记录: |