Blog Infos
Author
Published
Topics
,
Published
Topics
,
How to deserialize raw JSON objects with Moshi & Retrofit

You have probably come across different JSON structures in a single response. The ways in which you parse it in the application can be different. Imagine we have the following JSON:

[
{
"type": "TRUCK",
"waterCannon": true
},
{
"type": "PLANE",
"wingsSpanInMeters": 20
}
]

We can deserialize this JSON in different ways. A bad way to deserialize would be to put each property into a single object.

@JsonClass(generateAdapter = true)
data class Vehicle(
@Json(name = "type") val type: VehicleType,
@Json(name = "waterCannon") val waterCannon: Boolean?,
@Json(name = "wingsSpanInMetres") val wingsSpanInMetres: Int?
)

This is where we confuse ourselves and others with whom we share the code.

Why is this way bad?

The answer is simple. With this we set that the truck has a property that is related to the wingspan, which isn’t true.

This is where polymorphism comes in handy.

What is a polymorphism?

Polymorphism is the ability of an object to take on many forms.

In our example:

enum class VehicleType {
TRUCK,
PLANE;
}
// 1
sealed interface Vehicle {
val type: VehicleType
}
// 2
@JsonClass(generateAdapter = true)
data class Truck (
@Json(name = "waterCannon") val waterCannon: Boolean
): Vehicle {
override val type = VehicleType.TRUCK
}
// 3
@JsonClass(generateAdapter = true)
data class Plane (
@Json(name = "wingsSpanInMeters") val wingsSpanInMeters: Int
): Vehicle {
override val type = VehicleType.PLANE
}

In the first step we defined the Vehicle sealed interface with abstract parameter type.

What is Sealed interface?

Sealed interfaces represent restricted class hierarchies that provide more control over inheritance.

You can read more about them here.

In the second and third steps, we added the appropriate classes for the different types, as well as Moshi’s codegen annotations.

Now, we need to create a factory using PolymorphicJsonAdapterFactory.

What is PolymorphicJsonAdapterFactory?

A JsonAdapter factory for objects that include type information in the JSON. When decoding JSON Moshi uses this type information to determine which class to decode to. When encoding Moshi uses the object’s class to determine what type information to include.

val vehicleFactory = PolymorphicJsonAdapterFactory.of(Vehicle::class.java, "type")
.withSubtype(Truck::class.java, VehicleType.TRUCK.name)
.withSubtype(Plane::class.java, VehicleType.PLANE.name)

Job Offers

Job Offers

There are currently no vacancies.

OUR VIDEO RECOMMENDATION

, ,

Migrating to Jetpack Compose – an interop love story

Most of you are familiar with Jetpack Compose and its benefits. If you’re able to start anew and create a Compose-only app, you’re on the right track. But this talk might not be for you…
Watch Video

Migrating to Jetpack Compose - an interop love story

Simona Milanovic
Android DevRel Engineer for Jetpack Compose
Google

Migrating to Jetpack Compose - an interop love story

Simona Milanovic
Android DevRel Engin ...
Google

Migrating to Jetpack Compose - an interop love story

Simona Milanovic
Android DevRel Engineer f ...
Google

Jobs

Here we need to indicate which JSON property should be used to identify the specific class to be used. In our example it is type.

Finally, we need to add the factory to the Moshi builder.

val moshi = Moshi.Builder()
.add(vehicleFactory)
.build()
What if we get a non-enum type in the response?

There we have a problem, this factory will not work well then. One solution is to add UNKNOWN as a new type inside VehicleType and define a default value within the factory.

enum class VehicleType {
TRUCK,
PLANE,
UNKNOWN; //here
}
...
object Unknown: Vehicle {
override val type: VehicleType = VehicleType.UNKNOWN
}
val vehicleFactory = PolymorphicJsonAdapterFactory.of(Vehicle::class.java, "type")
.withSubtype(Truck::class.java, VehicleType.TRUCK.name)
.withSubtype(Plane::class.java, VehicleType.PLANE.name)
.withDefaultValue(Unknown) // here

Photo by Antonio Janeski on Unsplash

 

Thats it. I hope it was helpful. 🙂

This article was originally published on proandroiddev.com on July 09, 2022

YOU MAY BE INTERESTED IN

YOU MAY BE INTERESTED IN

blog
In this article, we will be discussing the process of refreshing JSON Web Tokens…
READ MORE
blog
In this article, we’ll be implementing JWT (JSON Web Token) authentication and silent refresh…
READ MORE

Leave a Reply

Your email address will not be published. Required fields are marked *

Fill out this field
Fill out this field
Please enter a valid email address.

Menu