7

我有代码片段

cursor.downField("params").downField("playlist").downField("items").as[List[Clip]]

其中 Clip 是字符串和数字的简单案例类。传入的 Json 应包含一个带有“项目”数组的 json 对象“播放列表”,其中每个项目都是一个剪辑。所以json应该看起来像

{
  "playlist": {
      "name": "Sample Playlist",
      "items": [
        {
          "clipId":"xyz", 
          "name":"abc"
        },
        {
          "clipId":"pqr", 
          "name":"def"
        } 
      ]
   }
}

使用上面的代码片段,我得到了编译错误:

 Error:(147, 81) could not find implicit value for parameter d:     
 io.circe.Decoder[List[com.packagename.model.Clip]]
      cursor.downField("params").downField("playlist").downField("items").as[List[Clip]]

我究竟做错了什么?如何使用 circe 为简单项目列表/数组设置解码?

4

3 回答 3

14

为了完整起见,您可以创建一个自定义解码器,在其处理中包含导航,而不是导航到 JSON 值然后解码剪辑:

import io.circe.Decoder, io.circe.generic.auto._

case class Clip(clipId: String, name: String)

val decodeClipsParam = Decoder[List[Clip]].prepare(
  _.downField("params").downField("playlist").downField("items")
)

然后如果你有这个:

val json = """{ "params": {
  "playlist": {
      "name": "Sample Playlist",
      "items": [
        {
          "clipId":"xyz", 
          "name":"abc"
        },
        {
          "clipId":"pqr", 
          "name":"def"
        } 
      ]
   }
}}"""

您可以像这样使用解码器:

scala> io.circe.parser.decode(json)(decodeClipsParam)
res3: Either[io.circe.Error,List[Clip]] = Right(List(Clip(xyz,abc), Clip(pqr,def)))

我可能会更进一步并使用自定义案例类:

import io.circe.generic.auto._
import io.circe.generic.semiauto.deriveDecoder

case class Clip(clipId: String, name: String)
case class PlaylistParam(name: String, items: List[Clip])

object PlaylistParam {
  implicit val decodePlaylistParam: Decoder[PlaylistParam] =
    deriveDecoder[PlaylistParam].prepare(
      _.downField("params").downField("playlist")
    )
}

现在你可以这样写:

scala> io.circe.parser.decode[PlaylistParam](json).foreach(println)
PlaylistParam(Sample Playlist,List(Clip(xyz,abc), Clip(pqr,def)))

不过,您想如何拆分导航和解码主要是个人喜好问题。

于 2017-02-17T18:19:05.603 回答
3

谢谢您的帮助。在离开一段时间并以新鲜的眼睛回来后,我能够弄清楚。

我认为我使用 downArray 函数出错了。

我的解决方案是执行以下操作:

override def main(args: Array[String]): Unit = {
    import ClipCodec.decodeClip

    val json = parse(Source.fromFile("playlist.json").mkString).right.get
    val clips = json.hcursor.downField("params").downField("playlist")
                   .downField("items").as[Seq[Clip]]

  }
于 2017-02-17T16:59:24.540 回答
1

Circe 正在寻找一个隐式声明的解码器,List[Clip]但找不到它。

我怀疑您没有手动或(半)自动定义解码器。您可以按照官方文档https://circe.github.io/circe/codec.html来完成这两项工作。

不幸的是,我无法提供比这更多的细节,因为这个问题相当模糊。当给出更多细节时,我会更新我的答案。

于 2017-02-17T16:08:35.163 回答