Skip to content
Advertisement

Dynamically choosing which class to create object of from json

I have an interesting problem that I’m having trouble coming up with a clean solution for. My application reads collections of json objects that it needs to deserialize to this or that class type based on a field in the json itself. I have no control over the json structure or how it gets to my application.

I’ve created models for each type of object that could be coming to the application and I’ve reached a point where I’m trying to build a service that pulls out the ‘type’ field and then uses ObjectMapper to deserialize the json to the appropriate Model.

Json example:

JavaScript

Models:

JavaScript

Service?:

JavaScript

I don’t want a massive switch statement that says “If type value is this then deserialize to that” but I’m struggling to come up with something clean that does that. I thought maybe a generic model class where the generic parameter is the Model Type and the only field is the instance of that model type but that doesn’t seem right either and I’m not sure what that buys me. I could also have some sort of empty abstract class that all of the models extend but that seems horrible too. How do I deal with this? Extra points for an example.

Advertisement

Answer

I use the concept of a parent interface Vehicle with 2 classes Car and Truck. In your case this means Model1 and Model2 should implement a common interface.

My test class:

JavaScript

Somewhere you will need to store a mapping between the value of the type field and the corresponding class. Depending on the location where you want this the implementation is different.

1) The parent type holds the list of subtypes:

JavaScript

The models for Car and Truck are simple POJO without any annotations:

JavaScript

2) A separate resolver holds the mapping:

Vehicle contains the extra annotation @JsonTypeIdResolver

JavaScript

The JsonResolver class holds the mapping between the type field value and the class:

JavaScript

3) The json contains the full class name:

If you accept your serialized json holds the full java class name, you don’t need a resolver but specify use = JsonTypeInfo.Id.CLASS:

JavaScript

Solution 3 is the easiest to implement but personally I don’t like the idea to have full java class names in my data. It can be a potential risk if you start to refactor your java packages.

User contributions licensed under: CC BY-SA
1 People found this is helpful
Advertisement