我有一个由本身需要序列化为 JSON 的类型键入的 Scala Map。由于 JSON 的性质要求对象的键名是字符串,因此无法直接进行简单的映射。
我希望实现的解决方法是在序列化为 JSON 之前将 Map 转换为 Set,然后在反序列化后从 Set 转换回 Map。
我知道在特定类型上使用密钥序列化程序的其他方法,例如Serialising Map with Jackson,但是,我需要一个适用于任意密钥类型的解决方案,在这方面,转换为 Set 并再次转换在我看来是最好的选择。
通过从下面的jackson-module-scala修改MapSerializerModule.scala ,我已经成功地将带有包装对象的 Set 序列化为 Set ,但我对 Jackson 的内部结构不够熟悉,无法让 JSON 反序列化回我开始的 Map和。
我应该补充一点,我控制序列化和反序列化方面,所以 JSON 看起来并不重要。
case class Wrapper[K, V](
value: Set[(K, V)]
)
class MapConverter[K, V](inputType: JavaType, config: SerializationConfig)
extends StdConverter[Map[K, V], Wrapper[K, V]] {
def convert(value: Map[K, V]): Wrapper[K, V] = {
val set = value.toSet
Wrapper(set)
}
override def getInputType(factory: TypeFactory) = inputType
override def getOutputType(factory: TypeFactory) =
factory.constructReferenceType(classOf[Wrapper[_, _]], inputType.getContentType)
.withTypeHandler(inputType.getTypeHandler)
.withValueHandler(inputType.getValueHandler)
}
object MapSerializerResolver extends Serializers.Base {
val MAP = classOf[Map[_, _]]
override def findMapLikeSerializer(
config: SerializationConfig,
typ: MapLikeType,
beanDesc: BeanDescription,
keySerializer: JsonSerializer[AnyRef],
elementTypeSerializer: TypeSerializer,
elementValueSerializer: JsonSerializer[AnyRef]): JsonSerializer[_] = {
val rawClass = typ.getRawClass
if (!MAP.isAssignableFrom(rawClass)) null
else new StdDelegatingSerializer(new MapConverter(typ, config))
}
}
object Main {
def main(args: Array[String]): Unit = {
val objMap = Map(
new Key("k1", "k2") -> "k1k2",
new Key("k2", "k3") -> "k2k3")
val om = new ObjectMapper()
om.registerModule(DefaultScalaModule)
om.registerModule(ConverterModule)
val res = om.writeValueAsString(objMap)
println(res)
}
}