Adding Custom Properties Using Jackson MixIns
Jackson is one of the standard solutions in Java world when you need to serialize/deserialize your classes. Not only it supports several different data formats, but it’s also very well designed, documented and easy to use. I’m particularly fond of mix-ins feature. Mix-ins provide a convenient way to customize class (de)serialization without modifying it.
Let me first show you a simple example how you can serialize your class to YAML format using Jackson. Consider the following code:
class MyClazz {
public int a = 10;
public int b = 15;
}
class Main {
public static void main(String[] args) throws JsonProcessingException {
YAMLFactory yamlFactory = new YAMLFactory();
ObjectMapper mapper = new ObjectMapper(yamlFactory);
System.out.println(mapper.writeValueAsString(new MyClazz()));
// Output:
//---
//a: 10
//b: 15
}
}
Let’s suppose that we can’t or don’t want to modify MyClazz
, but we want to skip b
during serialization.
We can do it by adding a mix-in for this class:
private abstract class MyClazzMixIn {
int a;
// Jackson annotation that tells serializer to skip field/method
@JsonIgnore
int b;
}
class Main {
public static void main(String[] args) throws JsonProcessingException {
YAMLFactory yamlFactory = new YAMLFactory();
ObjectMapper mapper = new ObjectMapper(yamlFactory);
mapper.addMixIn(MyClazz.class, MyClazzMixIn.class);
System.out.println(mapper.writeValueAsString(new MyClazz()));
// Output:
//---
//a: 10
}
}
Notice that we add @JsonIgnore annotation in mix-in class which duplicates MyClazz
definitions.
You actually don’t need to duplicate all the definitions, we can use more Jackson magic instead, but I don’t
want to go into the details.
Imagine now that you need to add custom information to your serialized object. In our example let’s add c
property which equals a + b
.
We can use JsonAppend annotation for this purpose. It allows us to add “virtual” properties to our objects.
Let’s rewrite our example:
@JsonAppend(props = {
@JsonAppend.Prop(value = MyWriter.class, name = "c", type = Integer.class)
})
private abstract static class MyClazzMixIn {
int a;
@JsonIgnore
int b;
}
public static void main(String[] args) throws JsonProcessingException {
YAMLFactory yamlFactory = new YAMLFactory();
ObjectMapper mapper = new ObjectMapper(yamlFactory);
mapper.addMixIn(MyClazz.class, MyClazzMixIn.class);
System.out.println(mapper.writeValueAsString(new MyClazz()));
// Output:
//---
//a: 10
//c: 25
}
We don’t even have to modify ObjectMapper
initialization! Our mix-in class still is the only
place containing all the information about MyClazz
serialization.
To complete this story I’ll show an implementation of MyWriter
. It’s just a class that knows
how to evaluate the value of our “virtual” property given an instance of MyClazz
:
private static class MyWriter extends VirtualBeanPropertyWriter {
public MyWriter() {
}
public MyWriter(BeanPropertyDefinition propDef,
Annotations contextAnnotations,
JavaType declaredType) {
super(propDef, contextAnnotations, declaredType);
}
@Override
protected Object value(Object bean, JsonGenerator gen, SerializerProvider prov) {
MyClazz myClazz = (MyClazz)bean;
return myClazz.a + myClazz.b;
}
@Override
public VirtualBeanPropertyWriter withConfig(MapperConfig<?> config,
AnnotatedClass declaringClass,
BeanPropertyDefinition propDef,
JavaType type) {
return new MyWriter(propDef, declaringClass.getAnnotations(), type);
}
}
Leave a comment