将JSON数组解组到案例类别列表时出错

时间:2018-09-23 05:49:44

标签: json scala circe

我一直在尝试使用circe将JSON从外部服务解组为案例类列表(我是circe和Scala的初学者)。

案例类如下:

case class User(
  id:            Int,
  username:      Option[String],
  permalink_url: Option[String],
  avatar_url:    Option[String],
  tracks:        Option[List[Track]],
  favorites:     Option[List[Track]],
  followings:    Option[List[User]] // The JSON below would populate this field
)

到目前为止,我对所有其他类型都使用了自动派生,但这是使用结构

   {
      "collection": [
          {
            "id": 42,
            "username": "user",
            "permalink_url": "foo://bar.baz",
            "followers_count": 15089,
            "followings_count": 498,
            "reposts_count": 31,
            "comments_count": 13,
            "online": false,
            "likes_count": 0,
            "playlist_count": 10
        },
        ...etc, etc
      ]
    }

为解决这个问题,我实现了一个自定义解码器:

implicit val followingsDecoder: Decoder[List[User]] = Decoder.instance( c =>
    for {
      collection <- c.downField("collection").as[List[User]]
    } yield collection
)

此操作失败并产生以下错误:

DecodingFailure(CNil, List(DownField(collection)))

我还尝试过使用circe的自动解码器派生,这会产生不同的错误:

DecodingFailure(CanBuildFrom for A, List())

我了解到circe的错误消息传递没有提供解码失败的字段。我不需要User case类中未指定的任何字段,并且不确定这是我接收的JSON还是我尝试对其进行解码的方式的问题。

这里缺少什么吗?我尝试了其他解码方式,例如半自动推导,以及尝试将列表解组为自己的案例类。

  implicit val followingsDecoder: Decoder[List[User]] = deriveDecoder[List[User]].prepare(
      _.downField("collection")
  )

我在这里错过了什么吗?如何将该JSON解析为案例类列表?

1 个答案:

答案 0 :(得分:0)

User的案例类定义是有效的,但是使用强制隐式followingsDecoder: Decoder来解组不是强制性的。 collection也可以包装到case class中,并且解码将自动完成。所以,如果我们有

case class UsersCollection(collection: List[User])

// I also added this one to make it compile:
case class Track(id: Int)

原始json,并添加了followings字段:

private val json =
  """
    |{
    |      "collection": [
    |          {
    |            "id": 42,
    |            "username": "user",
    |            "permalink_url": "foo://bar.baz",
    |            "followers_count": 15089,
    |            "followings_count": 498,
    |            "reposts_count": 31,
    |            "comments_count": 13,
    |            "online": false,
    |            "likes_count": 0,
    |            "playlist_count": 10,
    |            "followings": [
    |             { "id": 43 },
    |             { "id": 44 },
    |             { "id": 45 }
    |            ]
    |        }
    |      ]
    |    }
  """.stripMargin

它可以正确解析:

val decoded = decode[UsersCollection](json)
println(decoded)

// Right(UsersCollection(List(
//  User(42,Some(user),Some(foo://bar.baz),None,None,None,
//       Some(List(User(43,None,None,None,None,None,None),
//                 User(44,None,None,None,None,None,None),
//                 User(45,None,None,None,None,None,None)))))))

因此,如果您有可能,我建议添加另一个包装用户列表的临时案例类,而不要使用自定义解码器。