Custom JSON Deserialization With Jackson
Need to whip up a custom JSON deserializer with Jackson? Here's an awesome tutorial, with what to ignore, and how to create the deserializer.
Join the DZone community and get the full member experience.
Join For FreeWe consume a REST API as a JSON format and then unmarshal it to a POJO. Jackson’s org.codehaus.jackson.map.ObjectMapper “just works” out of the box and we really don’t do anything in most cases. But sometimes we need a custom deserializer to fulfill our custom needs and this tutorial will guide you through the process of creating your own custom deserializer.
Let’s say we have following entities:
public class User {
private Long id;
private String name;
private String email;
public Long getId() {
return id;
}
public User setId(Long id) {
this.id = id;
return this;
}
public String getName() {
return name;
}
public User setName(String name) {
this.name = name;
return this;
}
public String getEmail() {
return email;
}
public User setEmail(String email) {
this.email = email;
return this;
}
@Override
public String toString() {
final StringBuffer sb = new StringBuffer("User{");
sb.append("id=").append(id);
sb.append(", name='").append(name).append('\'');
sb.append(", email='").append(email).append('\'');
sb.append('}');
return sb.toString();
}
}
public class Program {
private Long id;
private String name;
private User createdBy;
private String contents;
public Program(Long id, String name, String contents, User createdBy) {
this.id = id;
this.name = name;
this.contents = contents;
this.createdBy = createdBy;
}
public Program() {}
public Long getId() {
return id;
}
public Program setId(Long id) {
this.id = id;
return this;
}
public String getName() {
return name;
}
public Program setName(String name) {
this.name = name;
return this;
}
public User getCreatedBy() {
return createdBy;
}
public Program setCreatedBy(User createdBy) {
this.createdBy = createdBy;
return this;
}
public String getContents() {
return contents;
}
public Program setContents(String contents) {
this.contents = contents;
return this;
}
@Override
public String toString() {
final StringBuffer sb = new StringBuffer("Program{");
sb.append("id=").append(id);
sb.append(", name='").append(name).append('\'');
sb.append(", createdBy=").append(createdBy);
sb.append(", contents='").append(contents).append('\'');
sb.append('}');
return sb.toString();
}
}
Let’s serialize/marshal an object first:
User user = new User();
user.setId(1L);
user.setEmail("example@example.com");
user.setName("Bazlur Rahman");
Program program = new Program();
program.setId(1L);
program.setName("Program @# 1");
program.setCreatedBy(user);
program.setContents("Some contents");
ObjectMapper objectMapper = new ObjectMapper();
final String json = objectMapper.writeValueAsString(program);
System.out.println(json);
The above code will produce the following JSON:
{
"id": 1,
"name": "Program @# 1",
"createdBy": {
"id": 1,
"name": "Bazlur Rahman",
"email": "example@example.com"
},
"contents": "Some contents"
}
Now you can do the opposite very easily. If we have this JSON, we can unmarshal to a program object using ObjectMapper
as follows:
String jsonString = "{\"id\":1,\"name\":\"Program @# 1\",\"createdBy\":{\"id\":1,\"name\":\"Bazlur Rahman\",\"email\":\"example@example.com\"},\"contents\":\"Some contents\"}";
final Program program1 = objectMapper.readValue(jsonString, Program.class);
System.out.println(program1);
Now let’s say, this is not the real case, we are going to have a different JSON from an API which doesn’t match with our program class:
{
"id": 1,
"name": "Program @# 1",
"ownerId": 1
"contents": "Some contents"
}
Look at the JSON string, you can see, it has a different field that is ownerId
.
Now if you want serialize this JSON as we did earlier, you will have exceptions.
There are two ways to avoid exceptions and have this seralized: ignore the unknown fields and write a custom deserializer.
Ignore the Unknown Fields
Ignore the ownrId
. Add following annotation in the Program class:
@JsonIgnoreProperties(ignoreUnknown = true)
public class Program {}
Write Custom Deserializer
But there are cases when you actually need this owerId
field. Let's say you want to relate it as an id of the user class.
In such a case, you need to write a custom deserializer:
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.core.ObjectCodec;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonDeserializer;
import com.fasterxml.jackson.databind.JsonNode;
import java.io.IOException;
public class ProgramDeserializer extends JsonDeserializer<Program> {
@Override
public Program deserialize(JsonParser jp, DeserializationContext ctxt) throws IOException, JsonProcessingException {
ObjectCodec oc = jp.getCodec();
JsonNode node = oc.readTree(jp);
final Long id = node.get("id").asLong();
final String name = node.get("name").asText();
final String contents = node.get("contents").asText();
final long ownerId = node.get("ownerId").asLong();
User user = new User();
user.setId(ownerId);
return new Program(id, name, contents, user);
}
}
As you can see, first you have to access the JsonNode from the JonsParser. And then you can easily extract information from a JsonNode using get method. and you have to be make sure about the field name. It should be the exact name, spelling mistake will cause exceptions.
And finally, you have to register your ProgramDeserializer to the ObjectMapper
.
ObjectMapper mapper = new ObjectMapper();
SimpleModule module = new SimpleModule();
module.addDeserializer(Program.class, new ProgramDeserializer());
mapper.registerModule(module);
String newJsonString = "{\"id\":1,\"name\":\"Program @# 1\",\"ownerId\":1,\"contents\":\"Some contents\"}";
final Program program2 = mapper.readValue(newJsonString, Program.class);
Alternatively, you can use annotation to register the deserializer directly:
@JsonDeserialize(using = ProgramDeserializer.<b>class</b>)
public class Program {}
Full source code can be found here.
Published at DZone with permission of A N M Bazlur Rahman, DZone MVB. See the original article here.
Opinions expressed by DZone contributors are their own.
Comments