JSON
Spring Boot 与以下 JSON 映射库集成:
-
Jackson 3
-
Jackson 2
-
Gson
-
JSON-B
-
Kotlin 序列化
Jackson 3 是首选和默认库。
支持 Jackson 2 的功能已弃用,并将在未来的 Spring Boot 4.x 版本中移除。 它仅用于简化从 Jackson 2 到 Jackson 3 的迁移,长期来看不应依赖于此。
Jackson 3
Jackson 3 的自动配置已提供,Jackson 是 spring-boot-starter-json 的一部分。
当 Jackson 在类路径上时,会自动配置一个 JsonMapper bean。
提供了多个配置属性用于 自定义 JsonMapper 的配置。
自定义序列化器和反序列化器
如果您使用 Jackson 来序列化和反序列化 JSON 数据,可能需要编写自己的 ValueSerializer 和 ValueDeserializer 类。
自定义序列化器通常通过模块向 Jackson 注册,但 Spring Boot 提供了一种替代方案——@JacksonComponent 注解,可更便捷地直接注册 Spring Bean。
您可以直接在 @JacksonComponent 注解上使用该注解,应用于 ValueSerializer、ValueDeserializer 或 KeyDeserializer 的实现类。
您还可以将其用于包含序列化器/反序列化器作为内部类的类,如下例所示:
-
Java
-
Kotlin
import tools.jackson.core.JsonGenerator;
import tools.jackson.core.JsonParser;
import tools.jackson.databind.DeserializationContext;
import tools.jackson.databind.JsonNode;
import tools.jackson.databind.SerializationContext;
import tools.jackson.databind.ValueDeserializer;
import tools.jackson.databind.ValueSerializer;
import org.springframework.boot.jackson.JacksonComponent;
@JacksonComponent
public class MyJacksonComponent {
public static class Serializer extends ValueSerializer<MyObject> {
@Override
public void serialize(MyObject value, JsonGenerator jgen, SerializationContext context) {
jgen.writeStartObject();
jgen.writeStringProperty("name", value.getName());
jgen.writeNumberProperty("age", value.getAge());
jgen.writeEndObject();
}
}
public static class Deserializer extends ValueDeserializer<MyObject> {
@Override
public MyObject deserialize(JsonParser jsonParser, DeserializationContext ctxt) {
JsonNode tree = jsonParser.readValueAsTree();
String name = tree.get("name").stringValue();
int age = tree.get("age").intValue();
return new MyObject(name, age);
}
}
}
import tools.jackson.core.JsonGenerator
import tools.jackson.core.JsonParser
import tools.jackson.databind.DeserializationContext
import tools.jackson.databind.JsonNode
import tools.jackson.databind.SerializationContext
import tools.jackson.databind.ValueDeserializer
import tools.jackson.databind.ValueSerializer
import org.springframework.boot.jackson.JacksonComponent
@JacksonComponent
class MyJacksonComponent {
class Serializer : ValueSerializer<MyObject>() {
override fun serialize(value: MyObject, jgen: JsonGenerator, serializers: SerializationContext) {
jgen.writeStartObject()
jgen.writeStringProperty("name", value.name)
jgen.writeNumberProperty("age", value.age)
jgen.writeEndObject()
}
}
class Deserializer : ValueDeserializer<MyObject>() {
override fun deserialize(jsonParser: JsonParser, ctxt: DeserializationContext): MyObject {
val tree = jsonParser.readValueAsTree<JsonNode>()
val name = tree["name"].stringValue()
val age = tree["age"].intValue()
return MyObject(name, age)
}
}
}
@JacksonComponent 中的所有 Bean 会自动向 Jackson 注册。
由于 @JacksonComponent 使用了 @Component 进行元注解,因此适用常规的组件扫描规则。
Spring Boot 还提供了 ObjectValueSerializer 和 ObjectValueDeserializer 两个基类,它们在序列化对象时为标准 Jackson 版本提供了有用的替代方案。
详情请参阅 API 文档中的 ObjectValueSerializer 和 ObjectValueDeserializer。
上述示例可重写为使用 ObjectValueSerializer 和 ObjectValueDeserializer,如下所示:
-
Java
-
Kotlin
import tools.jackson.core.JsonGenerator;
import tools.jackson.core.JsonParser;
import tools.jackson.databind.DeserializationContext;
import tools.jackson.databind.JsonNode;
import tools.jackson.databind.SerializationContext;
import org.springframework.boot.jackson.JacksonComponent;
import org.springframework.boot.jackson.ObjectValueDeserializer;
import org.springframework.boot.jackson.ObjectValueSerializer;
@JacksonComponent
public class MyJacksonComponent {
public static class Serializer extends ObjectValueSerializer<MyObject> {
@Override
protected void serializeObject(MyObject value, JsonGenerator jgen, SerializationContext context) {
jgen.writeStringProperty("name", value.getName());
jgen.writeNumberProperty("age", value.getAge());
}
}
public static class Deserializer extends ObjectValueDeserializer<MyObject> {
@Override
protected MyObject deserializeObject(JsonParser jsonParser, DeserializationContext context, JsonNode tree) {
String name = nullSafeValue(tree.get("name"), String.class);
int age = nullSafeValue(tree.get("age"), Integer.class);
return new MyObject(name, age);
}
}
}
import tools.jackson.core.JsonGenerator
import tools.jackson.core.JsonParser
import tools.jackson.databind.DeserializationContext
import tools.jackson.databind.JsonNode
import tools.jackson.databind.SerializationContext
import org.springframework.boot.jackson.JacksonComponent;
import org.springframework.boot.jackson.ObjectValueDeserializer
import org.springframework.boot.jackson.ObjectValueSerializer
@JacksonComponent
class MyJacksonComponent {
class Serializer : ObjectValueSerializer<MyObject>() {
override fun serializeObject(value: MyObject, jgen: JsonGenerator, context: SerializationContext) {
jgen.writeStringProperty("name", value.name)
jgen.writeNumberProperty("age", value.age)
}
}
class Deserializer : ObjectValueDeserializer<MyObject>() {
override fun deserializeObject(jsonParser: JsonParser, context: DeserializationContext,
tree: JsonNode): MyObject {
val name = nullSafeValue(tree["name"], String::class.java) ?: throw IllegalStateException("name is null")
val age = nullSafeValue(tree["age"], Int::class.java) ?: throw IllegalStateException("age is null")
return MyObject(name, age)
}
}
}
混入类
Jackson 支持混入(mixins),可用于将额外的注解添加到目标类已声明的注解中。
Spring Boot 的 Jackson 自动配置会扫描应用程序的包,查找使用 @JacksonMixin 注解的类,并将其注册到自动配置的 JsonMapper 中。
该注册操作由 Spring Boot 的 JacksonMixinModule 执行。
Jackson 2
为 Jackson 2 提供的已弃用自动配置由 spring-boot-jackson2 模块提供。
当此模块在类路径上时,会自动配置一个 ObjectMapper bean。
提供了多个 spring.jackson2.* 配置属性,用于自定义配置。
要获得更多的控制权,请定义一个或多个 Jackson2ObjectMapperBuilderCustomizer bean。
当同时存在 Jackson 3 和 Jackson 2 时,可以使用各种配置属性来表明优先使用 Jackson 2:
-
spring.graphql.rsocket.preferred-json-mapper -
spring.http.codecs.preferred-json-mapper(由 Spring WebFlux 和响应式 HTTP 客户端使用) -
spring.http.converters.preferred-json-mapper(由Spring MVC和命令式HTTP客户端使用) -
spring.rsocket.preferred-mapper -
spring.websocket.messaging.preferred-json-mapper
在每种情况下,将相关属性设置为jackson2,以表示优先使用Jackson 2。
Gson
提供了 Gson 的自动配置。
当类路径中包含 Gson 时,会自动配置一个 Gson 类型的 Bean。
提供了多个 spring.gson.* 配置属性,用于自定义配置。
如需更精细的控制,可以使用一个或多个 GsonBuilderCustomizer 类型的 Bean。
JSON-B
提供了对 JSON-B 的自动配置。
当类路径中存在 JSON-B API 及其实现时,将自动配置一个 Jsonb Bean。
首选的 JSON-B 实现是 Eclipse Yasson,其依赖管理已由框架提供。
Kotlin 序列化
为Kotlin序列化提供的自动配置。
当 kotlinx-serialization-json 在类路径上时,会自动配置一个 Json bean。
提供了几个 spring.kotlinx.serialization.json.* 配置属性,用于自定义配置。