在Play for Scala中序列化树

时间:2016-07-04 01:55:53

标签: scala playframework playframework-2.0

我有以下类是树节点:

case class Node[A] ( id: Int, data: A, var children: Option[Seq[Node[A]]],
                        var parent: Option[Node[A]] )

其中id是节点id号,data表示存储在节点中的信息,children是子节点列表,parent是父节点。

我想用树生成一个json,所以我编写了以下隐式val:

implicit val nodeWrite : Writes[Node[Data]] = (
      (JsPath \ "sk").write[Int] and
      (JsPath \ "dat").write[Data] and
      (JsPath \ "ch").write[Option[Seq[Node[Data]]]] and
      (JsPath \ "par").write[Option[Node[Data]]]
  ) (unlift(Node[Data].unapply))

然而编译器抱怨:

  

方法的缺少参数列表适用于对象节点未应用   方法仅在函数类型时转换为函数   预期。您可以通过编写apply _或来使此转换显式化   申请()而不是申请。

如何解决这个问题?

更新

数据定义为:

case class Data (descrip: String)

更新2

由于我需要一个具有N个根的树,我创建了一个包含节点序列的Tree类。

case class Tree[A] ( var nodes: Option[Seq[Node[A]]] )

但是我在序列化树时遇到问题:

  implicit val treeWrite : Writes[Tree[Data]] = new Writes[Tree[Data]] {
    def writes(x: Tree[Data]) = {
      Json.obj(
        "nodes" -> x.nodes.map(_.map(n => writes(n)))
        )
    }
  }

它抛出

  

类型不匹配;发现:选项[无需]:   play.api.libs.json.Json.JsValueWrapper

x.nodes.map行。

1 个答案:

答案 0 :(得分:1)

我没有完整的答案,但您可以通过指定类型来帮助编译:

(unlift[Node[Data],(Int, Data, Option[Seq[Node[Data]]], Option[Node[Data]])]
  (Node.unapply[Data](_)))

但它对你没有帮助,因为你必须使用lazyWrite的递归类型。我建议在这里使用更明确的方法:

implicit val nodeWrite : Writes[Node[Data]] = new Writes[Node[Data]] {
  def writes(x: Node[Data]) = {
    Json.obj(
      "id" -> x.id,
      "data" -> x.data,
      "children" -> x.children.map(_.map(n => writes(n))),
      "parent" -> x.parent.map(n => writes(n)))
  }
}
val child = Node(1, "child", None, None)
val node = Node(1, "data", Some(List(child)), None)

Json toJson node
res0: play.api.libs.json.JsValue = {"id":1,"data":"data",
   "children":[{"id":1,"data":"child","children":null,"parent":null}],"parent":null}

添加空处理,你会没事的。