3

我定义了一个具有“视频”信息的用户:

case class User(name:String, video: Option[Video])
case class Video(title:String, url:String)

我们有这样一个json:

{
   "name": "somename",
   "video": {
       "title": "my-video",
       "url": "https://youtube.com/watch?v=123123"
   }
}

我可以使用这样的代码来解析它:

implicit def DecodeUser: DecodeJson[User] = for {
    name <- as[String]("name")
    video <- as[Option[Video]]("video")
} yield User(name, video)

implicit def DecodeVideo: DecodeJson[Option[Video]] = for {
    titleOpt <- as[Option[String]]("title")
    urlOpt <- as[Option[String]]("url")
} yield (titleOpt, urlOpt) match {
   case (Some(title), Some(url)) => Video(title, url)
   case _ => None
}

从 中DecodeVideo,您可以看到我只想在同时提供“标题”和“url”的情况下提供视频。

如果 json 包含“视频”部分,它运行良好。但如果没有,argonaut 将报告未提供“视频”部分。

如何使“视频”可选?

4

1 回答 1

-1

我似乎无法弄清楚您的代码如何与 argonaut 集成。该方法的所有实例as[T]似乎都与您使用的签名不匹配。无论如何,这是一个类似的问题和解决方案:

object Test {

  case class Video(title: String, url: String)

  def test(titleOptIn: List[Option[String]], urlOptIn: List[Option[String]]): List[Option[Video]] = {
    for {
      titleOpt <- titleOptIn
      urlOpt <- urlOptIn
    } yield (titleOpt, urlOpt) match {
      case (Some(title), Some(url)) => Some(Video(title, url))
      case _ => None.asInstanceOf[Option[Video]]
    }
  }

  def main(args: Array[String]): Unit = {
    test(List(Some("t")), List(Some("u"), None)).foreach(println)
  }
}

// Has Output:
// Some(Video(t,u))
// None

特别注意,yield comprehension 应该返回 an ,Option[String]因为您的 yield 很可能将结果包装在DecodeJson就像我的示例将它包装在 a中一样List。注意asInstanceOfon None 是可选的;如果它不存在,IntelliJ 会抱怨,但它实际上编译得很好。

我认为您缺少的具体内容VideoSome.

于 2014-10-22T14:26:28.007 回答