circe如何将一个泛型类型的对象解析为Json?

sow*_*wen 4 scala

这是我想要做的

case class MessageModel (time: Long, content: String) {}
val message = MessageModel(123, "Hello World")
def jsonParser[A] (obj: A) : String = obj.asJson.noSpaces

println jsonParser[MessageModel](message)
Run Code Online (Sandbox Code Playgroud)

这不起作用,因为它会抱怨错误:(13,8)找不到参数编码器的隐含值:io.circe.Encoder [A] obj.asJson.noSpaces ^

我有点理解为什么会这样,但是有办法解决它吗?

谢谢

Tra*_*own 7

circe中的编码和解码由类型类提供,这意味着A如果要编码(或解码)类型的值,则必须能够在编译时证明您具有类型类实例A.

这意味着当你写这样的东西时:

import io.circe.syntax._

def jsonPrinter[A](obj: A): String = obj.asJson.noSpaces
Run Code Online (Sandbox Code Playgroud)

您没有提供有关Acirce的足够信息,以便能够打印该类型的值.您可以使用上下文绑定来解决此问题:

import io.circe.Encoder
import io.circe.syntax._

def jsonPrinter[A: Encoder](obj: A): String = obj.asJson.noSpaces
Run Code Online (Sandbox Code Playgroud)

Scala的语法糖就是这样的:

def jsonPrinter[A](obj: A)(implicit encoder: Encoder[A]): String =
  obj.asJson.noSpaces
Run Code Online (Sandbox Code Playgroud)

这两个都将编译,您可以向它们传递具有隐式Encoder实例的任何类型的值.为了您的MessageModel具体情况,你可以使用瑟茜的通用推导,因为它是一个案例类:

scala> import io.circe.generic.auto._
import io.circe.generic.auto._

scala> case class MessageModel(time: Long, content: String)
defined class MessageModel

scala> val message = MessageModel(123, "Hello World")
message: MessageModel = MessageModel(123,Hello World)

scala> jsonPrinter(message)
res0: String = {"time":123,"content":"Hello World"}
Run Code Online (Sandbox Code Playgroud)

请注意,如果没有auto导入,这将无法工作,导入Encoder为任何其成员都可编码的案例类(或密封的特征层次结构)提供实例.