【问题标题】:Serializer for interface / implementation接口/实现的序列化器
【发布时间】:2020-04-14 14:32:34
【问题描述】:

假设有一个接口和 2 个(或更多)实现:

interface IRunnable {
  fun run()
}

class Horse : IRunnable {
  override fun run() {    println("horse running")  }
}

class Dog : IRunnable {
  override fun run() {    println("dog running")  }
}

我想实现网络传输最简洁的JSON,即{"runnable":"H"}为Horse,{"runnable":"D"}为Dog

如果我无法修改接口和实现,我想将接口/实现序列化为 JSON ,根据 Kotlin 的文档,我必须写一个 custom serializer ,并使用 SerializersModule 来达到目标。

假设只有 HorseDog 实现 IRunnable ,这就是我所做的:

class RunnableSerializer : KSerializer<IRunnable> {
  override val descriptor: SerialDescriptor
    get() = StringDescriptor.withName("runnable")


  override fun serialize(encoder: Encoder, obj: IRunnable) {
    val stringValue = when (obj) {
      is Horse -> { "H" }
      is Dog -> { "D" }
      else -> { null }
    }
    stringValue?.also {
      encoder.encodeString(it)
    }
  }

  override fun deserialize(decoder: Decoder): IRunnable {
    return decoder.decodeString().let { value ->
      when(value) {
        "H" -> Horse()
        "D" -> Dog()
        else -> throw RuntimeException("invalid $value")
      }
    }
  }
}

但是当我尝试将 Horse 转换为 JSON ...

class RunnableTest {

  private val logger = KotlinLogging.logger { }

  @ImplicitReflectionSerializer
  @Test
  fun testJson() {
    val module1 = serializersModuleOf(IRunnable::class, RunnableSerializer())

    Json(context = module1).stringify(IRunnable::class.serializer(), Horse()).also {
      logger.info("json = {}", it)
    }
  }
}

输出错误:

kotlinx.serialization.SerializationException: 
Can't locate argument-less serializer for class IRunnable. For generic classes, such as lists, please provide serializer explicitly.

如何实现类似 {"runnable":"H"}{"runnable":"D"} 在这种情况下?

谢谢。

环境:

<kotlin.version>1.3.60</kotlin.version>
<serialization.version>0.14.0</serialization.version>

更新,完整的错误信息:

kotlinx.serialization.SerializationException: Can't locate argument-less serializer for class IRunnable. For generic classes, such as lists, please provide serializer explicitly.

    at kotlinx.serialization.PlatformUtilsKt.serializer(PlatformUtils.kt:12)
    at RunnableTest.testJson(RunnableTest.kt:17)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:498)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
    at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
    at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
    at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
    at com.intellij.rt.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:33)
    at com.intellij.rt.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:230)
    at com.intellij.rt.junit.JUnitStarter.main(JUnitStarter.java:58)

【问题讨论】:

  • 似乎HorseDog 类中的一个/两个都有参数化的构造函数(可能是可选的但参数化的)。
  • 不...这是完整的源代码,没有其他构造函数..
  • 这是完整的错误/堆栈跟踪吗?如果没有,您可以添加完整的跟踪吗?
  • 嗨,这是源代码和测试代码。 github.com/smallufo/kotlinPlay/tree/master/serialize/src。 ...
  • 我在询问错误的堆栈跟踪,因为源并不大,就像你之前说的那样。

标签: kotlin kotlinx.serialization


【解决方案1】:

您要实现的目标称为多态序列化which kotlinx.serialization supports。简而言之:它要求您将IRunnable 的所有派生类型注册为SerialModule 中的多态。对于您的代码,它应该如下所示:

val serialModule = SerializersModule {
    polymorphic(IRunnable::class) {
        Horse::class with HorseSerializer
        Dog::class with DogSerializer
    }
}

目前,您只注册IRunnable,正如异常正确表明的那样,无法初始化。你将如何初始化一个接口?你不能。相反,您想知道要初始化哪个特定的派生类型。这需要在 JSON 数据中嵌入类型信息,这就是多态序列化所做的。

此外,您不必实现自定义序列化程序。根据您链接到的文档,您可以创建external serializers for library classes。以下内容可能就足够了,因为 kotlinx.serialization 编译器插件会尝试为您生成合适的序列化程序,类似于直接在类上添加 @Serializable

@Serializer(forClass = Horse::class)
object HorseSerializer {}

@Serializer(forClass = Dog::class)
object DogSerializer {}

【讨论】:

  • 嗨,我试过你的代码,但还是不行...这里的代码:gist.github.com/smallufo/85632e827432a03e2da3f7d858d39460
  • 我发现文档意味着这种情况:gist.github.com/smallufo/c47e641cae24252b610e5d581b6cff78,接口必须在@Serializable容器内,在本例中为Zoo。然后它工作正常。但这不是我真正想要的......我希望一个简单的 {"runnable":"H"} 和 {"runnable":"D"} 映射
  • 如果您想指定自定义序列化(您提供的简单示例是自定义多态序列化),您需要实现一个自定义序列化程序,就像您最初尝试的那样。然后,您应该使用@Serializer(forClass) 属性注释您的自定义序列化程序,并在您想要进行自定义序列化的任何位置指定Serializable(with)。我不认为涉及串行模块。
  • 或者,注册为我在此答案中指定的多态,但为所有派生类型关联您的自定义序列化程序,而不是为每个特定类型生成一个插件。
猜你喜欢
  • 1970-01-01
  • 2014-05-23
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2023-04-07
相关资源
最近更新 更多