Usando @JsonComponent no Spring Boot
1. Visão geral
Este artigo rápido se concentra em como usar a anotação@JsonComponent no Spring Boot.
A anotação nos permite expor uma classe anotada para ser um serializador e / ou desserializador Jackson sem a necessidade de adicioná-lo aoObjectMapper manualmente.
Isso faz parte do módulo principal do Spring Boot, portanto, não há dependências adicionais necessárias em um aplicativo simples do Spring Boot.
2. Serialização
Vamos começar com o seguinte objetoUser contendo uma cor favorita:
public class User {
private Color favoriteColor;
// standard getters/constructors
}
Se serializarmos esse objeto usando Jackson com as configurações padrão, obteremos:
{ "favoriteColor": { "red": 0.9411764740943909, "green": 0.9725490212440491, "blue": 1.0, "opacity": 1.0, "opaque": true, "hue": 208.00000000000003, "saturation": 0.05882352590560913, "brightness": 1.0 } }
Podemos tornar o JSON muito mais condensado e legível apenas imprimindo os valores RGB - por exemplo, para serem usados em CSS.
Nesse sentido, só precisamos criar uma classe que implementeJsonSerializer:
@JsonComponent
public class UserJsonSerializer extends JsonSerializer {
@Override
public void serialize(User user, JsonGenerator jsonGenerator,
SerializerProvider serializerProvider) throws IOException,
JsonProcessingException {
jsonGenerator.writeStartObject();
jsonGenerator.writeStringField(
"favoriteColor",
getColorAsWebColor(user.getFavoriteColor()));
jsonGenerator.writeEndObject();
}
private static String getColorAsWebColor(Color color) {
int r = (int) Math.round(color.getRed() * 255.0);
int g = (int) Math.round(color.getGreen() * 255.0);
int b = (int) Math.round(color.getBlue() * 255.0);
return String.format("#%02x%02x%02x", r, g, b);
}
}
Com este serializador, o JSON resultante foi reduzido para:
{"favoriteColor":"#f0f8ff"}
Devido à anotação@JsonComponent, o serializador é registrado no JacksonObjectMapper no aplicativo Spring Boot. Podemos testar isso com o seguinte teste JUnit:
@JsonTest
@RunWith(SpringRunner.class)
public class UserJsonSerializerTest {
@Autowired
private ObjectMapper objectMapper;
@Test
public void testSerialization() throws JsonProcessingException {
User user = new User(Color.ALICEBLUE);
String json = objectMapper.writeValueAsString(user);
assertEquals("{\"favoriteColor\":\"#f0f8ff\"}", json);
}
}
3. Desserialização
Continuando com o mesmo exemplo, podemos escrever um desserializador que tornará a cor da webString em um objeto JavaFX Color:
@JsonComponent
public class UserJsonDeserializer extends JsonDeserializer {
@Override
public User deserialize(JsonParser jsonParser,
DeserializationContext deserializationContext) throws IOException,
JsonProcessingException {
TreeNode treeNode = jsonParser.getCodec().readTree(jsonParser);
TextNode favoriteColor
= (TextNode) treeNode.get("favoriteColor");
return new User(Color.web(favoriteColor.asText()));
}
}
Vamos testar o novo desserializador e garantir que tudo funcione conforme o esperado:
@JsonTest
@RunWith(SpringRunner.class)
public class UserJsonDeserializerTest {
@Autowired
private ObjectMapper objectMapper;
@Test
public void testDeserialize() throws IOException {
String json = "{\"favoriteColor\":\"#f0f8ff\"}"
User user = objectMapper.readValue(json, User.class);
assertEquals(Color.ALICEBLUE, user.getFavoriteColor());
}
}
4. Serializer eDeserializer em uma classe
Quando desejado, podemos conectar o serializador e o desserializador em uma classe usando duas classes internas e adicionando@JsonComponent na classe envolvente:
@JsonComponent
public class UserCombinedSerializer {
public static class UserJsonSerializer
extends JsonSerializer {
@Override
public void serialize(User user, JsonGenerator jsonGenerator,
SerializerProvider serializerProvider) throws IOException,
JsonProcessingException {
jsonGenerator.writeStartObject();
jsonGenerator.writeStringField(
"favoriteColor", getColorAsWebColor(user.getFavoriteColor()));
jsonGenerator.writeEndObject();
}
private static String getColorAsWebColor(Color color) {
int r = (int) Math.round(color.getRed() * 255.0);
int g = (int) Math.round(color.getGreen() * 255.0);
int b = (int) Math.round(color.getBlue() * 255.0);
return String.format("#%02x%02x%02x", r, g, b);
}
}
public static class UserJsonDeserializer
extends JsonDeserializer {
@Override
public User deserialize(JsonParser jsonParser,
DeserializationContext deserializationContext)
throws IOException, JsonProcessingException {
TreeNode treeNode = jsonParser.getCodec().readTree(jsonParser);
TextNode favoriteColor = (TextNode) treeNode.get(
"favoriteColor");
return new User(Color.web(favoriteColor.asText()));
}
}
}
5. Conclusão
Este rápido tutorial mostrou como adicionar rapidamente um serializador / desserializador Jackson em um aplicativo Spring Boot aproveitando a varredura de componente com a anotação@JsonComponent.
Os trechos de código podem ser encontradosover on GitHub.