我有以下类是树节点:
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
行。
答案 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}
添加空处理,你会没事的。